diff options
Diffstat (limited to 'net/ipv6/addrconf.c')
| -rw-r--r-- | net/ipv6/addrconf.c | 2519 | 
1 files changed, 1583 insertions, 936 deletions
diff --git a/net/ipv6/addrconf.c b/net/ipv6/addrconf.c index 1023ad0d2b1..5667b3003af 100644 --- a/net/ipv6/addrconf.c +++ b/net/ipv6/addrconf.c @@ -38,6 +38,8 @@   *						status etc.   */ +#define pr_fmt(fmt) "IPv6: " fmt +  #include <linux/errno.h>  #include <linux/types.h>  #include <linux/kernel.h> @@ -61,11 +63,14 @@  #include <linux/delay.h>  #include <linux/notifier.h>  #include <linux/string.h> +#include <linux/hash.h>  #include <net/net_namespace.h>  #include <net/sock.h>  #include <net/snmp.h> +#include <net/af_ieee802154.h> +#include <net/firewire.h>  #include <net/ipv6.h>  #include <net/protocol.h>  #include <net/ndisc.h> @@ -77,24 +82,22 @@  #include <net/pkt_sched.h>  #include <linux/if_tunnel.h>  #include <linux/rtnetlink.h> - -#ifdef CONFIG_IPV6_PRIVACY +#include <linux/netconf.h>  #include <linux/random.h> -#endif -  #include <linux/uaccess.h>  #include <asm/unaligned.h>  #include <linux/proc_fs.h>  #include <linux/seq_file.h> +#include <linux/export.h>  /* Set to 3 to get tracing... */  #define ACONF_DEBUG 2  #if ACONF_DEBUG >= 3 -#define ADBG(x) printk x +#define ADBG(fmt, ...) printk(fmt, ##__VA_ARGS__)  #else -#define ADBG(x) +#define ADBG(fmt, ...) do { if (0) printk(fmt, ##__VA_ARGS__); } while (0)  #endif  #define	INFINITY_LIFE_TIME	0xFFFFFFFF @@ -104,10 +107,6 @@ static inline u32 cstamp_delta(unsigned long cstamp)  	return (cstamp - INITIAL_JIFFIES) * 100UL / HZ;  } -#define ADDRCONF_TIMER_FUZZ_MINUS	(HZ > 50 ? HZ/50 : 1) -#define ADDRCONF_TIMER_FUZZ		(HZ / 4) -#define ADDRCONF_TIMER_FUZZ_MAX		(HZ) -  #ifdef CONFIG_SYSCTL  static void addrconf_sysctl_register(struct inet6_dev *idev);  static void addrconf_sysctl_unregister(struct inet6_dev *idev); @@ -121,11 +120,9 @@ static inline void addrconf_sysctl_unregister(struct inet6_dev *idev)  }  #endif -#ifdef CONFIG_IPV6_PRIVACY -static int __ipv6_regen_rndid(struct inet6_dev *idev); -static int __ipv6_try_regen_rndid(struct inet6_dev *idev, struct in6_addr *tmpaddr); +static void __ipv6_regen_rndid(struct inet6_dev *idev); +static void __ipv6_try_regen_rndid(struct inet6_dev *idev, struct in6_addr *tmpaddr);  static void ipv6_regen_rndid(unsigned long data); -#endif  static int ipv6_generate_eui64(u8 *eui, struct net_device *dev);  static int ipv6_count_addresses(struct inet6_dev *idev); @@ -136,10 +133,12 @@ static int ipv6_count_addresses(struct inet6_dev *idev);  static struct hlist_head inet6_addr_lst[IN6_ADDR_HSIZE];  static DEFINE_SPINLOCK(addrconf_hash_lock); -static void addrconf_verify(unsigned long); +static void addrconf_verify(void); +static void addrconf_verify_rtnl(void); +static void addrconf_verify_work(struct work_struct *); -static DEFINE_TIMER(addr_chk_timer, addrconf_verify, 0, 0); -static DEFINE_SPINLOCK(addrconf_verify_lock); +static struct workqueue_struct *addrconf_wq; +static DECLARE_DELAYED_WORK(addr_chk_work, addrconf_verify_work);  static void addrconf_join_anycast(struct inet6_ifaddr *ifp);  static void addrconf_leave_anycast(struct inet6_ifaddr *ifp); @@ -148,8 +147,13 @@ static void addrconf_type_change(struct net_device *dev,  				 unsigned long event);  static int addrconf_ifdown(struct net_device *dev, int how); -static void addrconf_dad_start(struct inet6_ifaddr *ifp, u32 flags); -static void addrconf_dad_timer(unsigned long data); +static struct rt6_info *addrconf_get_prefix_route(const struct in6_addr *pfx, +						  int plen, +						  const struct net_device *dev, +						  u32 flags, u32 noflags); + +static void addrconf_dad_start(struct inet6_ifaddr *ifp); +static void addrconf_dad_work(struct work_struct *w);  static void addrconf_dad_completed(struct inet6_ifaddr *ifp);  static void addrconf_dad_run(struct inet6_dev *idev);  static void addrconf_rs_timer(unsigned long data); @@ -161,8 +165,6 @@ static void inet6_prefix_notify(int event, struct inet6_dev *idev,  static bool ipv6_chk_same_addr(struct net *net, const struct in6_addr *addr,  			       struct net_device *dev); -static ATOMIC_NOTIFIER_HEAD(inet6addr_chain); -  static struct ipv6_devconf ipv6_devconf __read_mostly = {  	.forwarding		= 0,  	.hop_limit		= IPV6_DEFAULT_HOPLIMIT, @@ -171,17 +173,17 @@ static struct ipv6_devconf ipv6_devconf __read_mostly = {  	.accept_redirects	= 1,  	.autoconf		= 1,  	.force_mld_version	= 0, +	.mldv1_unsolicited_report_interval = 10 * HZ, +	.mldv2_unsolicited_report_interval = HZ,  	.dad_transmits		= 1,  	.rtr_solicits		= MAX_RTR_SOLICITATIONS,  	.rtr_solicit_interval	= RTR_SOLICITATION_INTERVAL,  	.rtr_solicit_delay	= MAX_RTR_SOLICITATION_DELAY, -#ifdef CONFIG_IPV6_PRIVACY  	.use_tempaddr 		= 0,  	.temp_valid_lft		= TEMP_VALID_LIFETIME,  	.temp_prefered_lft	= TEMP_PREFERRED_LIFETIME,  	.regen_max_retry	= REGEN_MAX_RETRY,  	.max_desync_factor	= MAX_DESYNC_FACTOR, -#endif  	.max_addresses		= IPV6_MAX_ADDRESSES,  	.accept_ra_defrtr	= 1,  	.accept_ra_pinfo	= 1, @@ -196,6 +198,7 @@ static struct ipv6_devconf ipv6_devconf __read_mostly = {  	.accept_source_route	= 0,	/* we do not accept RH0 by default. */  	.disable_ipv6		= 0,  	.accept_dad		= 1, +	.suppress_frag_ndisc	= 1,  };  static struct ipv6_devconf ipv6_devconf_dflt __read_mostly = { @@ -205,17 +208,18 @@ static struct ipv6_devconf ipv6_devconf_dflt __read_mostly = {  	.accept_ra		= 1,  	.accept_redirects	= 1,  	.autoconf		= 1, +	.force_mld_version	= 0, +	.mldv1_unsolicited_report_interval = 10 * HZ, +	.mldv2_unsolicited_report_interval = HZ,  	.dad_transmits		= 1,  	.rtr_solicits		= MAX_RTR_SOLICITATIONS,  	.rtr_solicit_interval	= RTR_SOLICITATION_INTERVAL,  	.rtr_solicit_delay	= MAX_RTR_SOLICITATION_DELAY, -#ifdef CONFIG_IPV6_PRIVACY  	.use_tempaddr		= 0,  	.temp_valid_lft		= TEMP_VALID_LIFETIME,  	.temp_prefered_lft	= TEMP_PREFERRED_LIFETIME,  	.regen_max_retry	= REGEN_MAX_RETRY,  	.max_desync_factor	= MAX_DESYNC_FACTOR, -#endif  	.max_addresses		= IPV6_MAX_ADDRESSES,  	.accept_ra_defrtr	= 1,  	.accept_ra_pinfo	= 1, @@ -230,121 +234,78 @@ static struct ipv6_devconf ipv6_devconf_dflt __read_mostly = {  	.accept_source_route	= 0,	/* we do not accept RH0 by default. */  	.disable_ipv6		= 0,  	.accept_dad		= 1, +	.suppress_frag_ndisc	= 1,  }; -/* IPv6 Wildcard Address and Loopback Address defined by RFC2553 */ -const struct in6_addr in6addr_any = IN6ADDR_ANY_INIT; -const struct in6_addr in6addr_loopback = IN6ADDR_LOOPBACK_INIT; -const struct in6_addr in6addr_linklocal_allnodes = IN6ADDR_LINKLOCAL_ALLNODES_INIT; -const struct in6_addr in6addr_linklocal_allrouters = IN6ADDR_LINKLOCAL_ALLROUTERS_INIT; -  /* Check if a valid qdisc is available */  static inline bool addrconf_qdisc_ok(const struct net_device *dev)  {  	return !qdisc_tx_is_noop(dev);  } -/* Check if a route is valid prefix route */ -static inline int addrconf_is_prefix_route(const struct rt6_info *rt) +static void addrconf_del_rs_timer(struct inet6_dev *idev)  { -	return (rt->rt6i_flags & (RTF_GATEWAY | RTF_DEFAULT)) == 0; +	if (del_timer(&idev->rs_timer)) +		__in6_dev_put(idev);  } -static void addrconf_del_timer(struct inet6_ifaddr *ifp) +static void addrconf_del_dad_work(struct inet6_ifaddr *ifp)  { -	if (del_timer(&ifp->timer)) +	if (cancel_delayed_work(&ifp->dad_work))  		__in6_ifa_put(ifp);  } -enum addrconf_timer_t { -	AC_NONE, -	AC_DAD, -	AC_RS, -}; +static void addrconf_mod_rs_timer(struct inet6_dev *idev, +				  unsigned long when) +{ +	if (!timer_pending(&idev->rs_timer)) +		in6_dev_hold(idev); +	mod_timer(&idev->rs_timer, jiffies + when); +} -static void addrconf_mod_timer(struct inet6_ifaddr *ifp, -			       enum addrconf_timer_t what, -			       unsigned long when) +static void addrconf_mod_dad_work(struct inet6_ifaddr *ifp, +				   unsigned long delay)  { -	if (!del_timer(&ifp->timer)) +	if (!delayed_work_pending(&ifp->dad_work))  		in6_ifa_hold(ifp); - -	switch (what) { -	case AC_DAD: -		ifp->timer.function = addrconf_dad_timer; -		break; -	case AC_RS: -		ifp->timer.function = addrconf_rs_timer; -		break; -	default: -		break; -	} -	ifp->timer.expires = jiffies + when; -	add_timer(&ifp->timer); +	mod_delayed_work(addrconf_wq, &ifp->dad_work, delay);  }  static int snmp6_alloc_dev(struct inet6_dev *idev)  { -	if (snmp_mib_init((void __percpu **)idev->stats.ipv6, -			  sizeof(struct ipstats_mib), -			  __alignof__(struct ipstats_mib)) < 0) +	int i; + +	idev->stats.ipv6 = alloc_percpu(struct ipstats_mib); +	if (!idev->stats.ipv6)  		goto err_ip; -	if (snmp_mib_init((void __percpu **)idev->stats.icmpv6, -			  sizeof(struct icmpv6_mib), -			  __alignof__(struct icmpv6_mib)) < 0) + +	for_each_possible_cpu(i) { +		struct ipstats_mib *addrconf_stats; +		addrconf_stats = per_cpu_ptr(idev->stats.ipv6, i); +		u64_stats_init(&addrconf_stats->syncp); +	} + + +	idev->stats.icmpv6dev = kzalloc(sizeof(struct icmpv6_mib_device), +					GFP_KERNEL); +	if (!idev->stats.icmpv6dev)  		goto err_icmp; -	if (snmp_mib_init((void __percpu **)idev->stats.icmpv6msg, -			  sizeof(struct icmpv6msg_mib), -			  __alignof__(struct icmpv6msg_mib)) < 0) +	idev->stats.icmpv6msgdev = kzalloc(sizeof(struct icmpv6msg_mib_device), +					   GFP_KERNEL); +	if (!idev->stats.icmpv6msgdev)  		goto err_icmpmsg;  	return 0;  err_icmpmsg: -	snmp_mib_free((void __percpu **)idev->stats.icmpv6); +	kfree(idev->stats.icmpv6dev);  err_icmp: -	snmp_mib_free((void __percpu **)idev->stats.ipv6); +	free_percpu(idev->stats.ipv6);  err_ip:  	return -ENOMEM;  } -static void snmp6_free_dev(struct inet6_dev *idev) -{ -	snmp_mib_free((void __percpu **)idev->stats.icmpv6msg); -	snmp_mib_free((void __percpu **)idev->stats.icmpv6); -	snmp_mib_free((void __percpu **)idev->stats.ipv6); -} - -/* Nobody refers to this device, we may destroy it. */ - -static void in6_dev_finish_destroy_rcu(struct rcu_head *head) -{ -	struct inet6_dev *idev = container_of(head, struct inet6_dev, rcu); -	kfree(idev); -} - -void in6_dev_finish_destroy(struct inet6_dev *idev) -{ -	struct net_device *dev = idev->dev; - -	WARN_ON(!list_empty(&idev->addr_list)); -	WARN_ON(idev->mc_list != NULL); - -#ifdef NET_REFCNT_DEBUG -	printk(KERN_DEBUG "in6_dev_finish_destroy: %s\n", dev ? dev->name : "NIL"); -#endif -	dev_put(dev); -	if (!idev->dead) { -		pr_warning("Freeing alive inet6 device %p\n", idev); -		return; -	} -	snmp6_free_dev(idev); -	call_rcu(&idev->rcu, in6_dev_finish_destroy_rcu); -} - -EXPORT_SYMBOL(in6_dev_finish_destroy); - -static struct inet6_dev * ipv6_add_dev(struct net_device *dev) +static struct inet6_dev *ipv6_add_dev(struct net_device *dev)  {  	struct inet6_dev *ndev; @@ -361,7 +322,8 @@ static struct inet6_dev * ipv6_add_dev(struct net_device *dev)  	rwlock_init(&ndev->lock);  	ndev->dev = dev;  	INIT_LIST_HEAD(&ndev->addr_list); - +	setup_timer(&ndev->rs_timer, addrconf_rs_timer, +		    (unsigned long)ndev);  	memcpy(&ndev->cnf, dev_net(dev)->ipv6.devconf_dflt, sizeof(ndev->cnf));  	ndev->cnf.mtu6 = dev->mtu;  	ndev->cnf.sysctl = NULL; @@ -376,19 +338,19 @@ static struct inet6_dev * ipv6_add_dev(struct net_device *dev)  	dev_hold(dev);  	if (snmp6_alloc_dev(ndev) < 0) { -		ADBG((KERN_WARNING -			"%s(): cannot allocate memory for statistics; dev=%s.\n", -			__func__, dev->name)); +		ADBG(KERN_WARNING +			"%s: cannot allocate memory for statistics; dev=%s.\n", +			__func__, dev->name);  		neigh_parms_release(&nd_tbl, ndev->nd_parms); -		ndev->dead = 1; -		in6_dev_finish_destroy(ndev); +		dev_put(dev); +		kfree(ndev);  		return NULL;  	}  	if (snmp6_register_dev(ndev) < 0) { -		ADBG((KERN_WARNING -			"%s(): cannot create /proc/net/dev_snmp6/%s\n", -			__func__, dev->name)); +		ADBG(KERN_WARNING +			"%s: cannot create /proc/net/dev_snmp6/%s\n", +			__func__, dev->name);  		neigh_parms_release(&nd_tbl, ndev->nd_parms);  		ndev->dead = 1;  		in6_dev_finish_destroy(ndev); @@ -403,16 +365,13 @@ static struct inet6_dev * ipv6_add_dev(struct net_device *dev)  	if (dev->flags & (IFF_NOARP | IFF_LOOPBACK))  		ndev->cnf.accept_dad = -1; -#if defined(CONFIG_IPV6_SIT) || defined(CONFIG_IPV6_SIT_MODULE) +#if IS_ENABLED(CONFIG_IPV6_SIT)  	if (dev->type == ARPHRD_SIT && (dev->priv_flags & IFF_ISATAP)) { -		printk(KERN_INFO -		       "%s: Disabled Multicast RS\n", -		       dev->name); +		pr_info("%s: Disabled Multicast RS\n", dev->name);  		ndev->cnf.rtr_solicits = 0;  	}  #endif -#ifdef CONFIG_IPV6_PRIVACY  	INIT_LIST_HEAD(&ndev->tempaddr_list);  	setup_timer(&ndev->regen_timer, ipv6_regen_rndid, (unsigned long)ndev);  	if ((dev->flags&IFF_LOOPBACK) || @@ -420,15 +379,13 @@ static struct inet6_dev * ipv6_add_dev(struct net_device *dev)  	    dev->type == ARPHRD_TUNNEL6 ||  	    dev->type == ARPHRD_SIT ||  	    dev->type == ARPHRD_NONE) { -		printk(KERN_INFO -		       "%s: Disabled Privacy Extensions\n", -		       dev->name);  		ndev->cnf.use_tempaddr = -1;  	} else {  		in6_dev_hold(ndev);  		ipv6_regen_rndid((unsigned long) ndev);  	} -#endif + +	ndev->token = in6addr_any;  	if (netif_running(dev) && addrconf_qdisc_ok(dev))  		ndev->if_flags |= IF_READY; @@ -439,13 +396,20 @@ static struct inet6_dev * ipv6_add_dev(struct net_device *dev)  	/* protected by rtnl_lock */  	rcu_assign_pointer(dev->ip6_ptr, ndev); +	/* Join interface-local all-node multicast group */ +	ipv6_dev_mc_inc(dev, &in6addr_interfacelocal_allnodes); +  	/* Join all-node multicast group */  	ipv6_dev_mc_inc(dev, &in6addr_linklocal_allnodes); +	/* Join all-router multicast group if forwarding is set */ +	if (ndev->cnf.forwarding && (dev->flags & IFF_MULTICAST)) +		ipv6_dev_mc_inc(dev, &in6addr_linklocal_allrouters); +  	return ndev;  } -static struct inet6_dev * ipv6_find_idev(struct net_device *dev) +static struct inet6_dev *ipv6_find_idev(struct net_device *dev)  {  	struct inet6_dev *idev; @@ -463,6 +427,226 @@ static struct inet6_dev * ipv6_find_idev(struct net_device *dev)  	return idev;  } +static int inet6_netconf_msgsize_devconf(int type) +{ +	int size =  NLMSG_ALIGN(sizeof(struct netconfmsg)) +		    + nla_total_size(4);	/* NETCONFA_IFINDEX */ + +	/* type -1 is used for ALL */ +	if (type == -1 || type == NETCONFA_FORWARDING) +		size += nla_total_size(4); +#ifdef CONFIG_IPV6_MROUTE +	if (type == -1 || type == NETCONFA_MC_FORWARDING) +		size += nla_total_size(4); +#endif +	if (type == -1 || type == NETCONFA_PROXY_NEIGH) +		size += nla_total_size(4); + +	return size; +} + +static int inet6_netconf_fill_devconf(struct sk_buff *skb, int ifindex, +				      struct ipv6_devconf *devconf, u32 portid, +				      u32 seq, int event, unsigned int flags, +				      int type) +{ +	struct nlmsghdr  *nlh; +	struct netconfmsg *ncm; + +	nlh = nlmsg_put(skb, portid, seq, event, sizeof(struct netconfmsg), +			flags); +	if (nlh == NULL) +		return -EMSGSIZE; + +	ncm = nlmsg_data(nlh); +	ncm->ncm_family = AF_INET6; + +	if (nla_put_s32(skb, NETCONFA_IFINDEX, ifindex) < 0) +		goto nla_put_failure; + +	/* type -1 is used for ALL */ +	if ((type == -1 || type == NETCONFA_FORWARDING) && +	    nla_put_s32(skb, NETCONFA_FORWARDING, devconf->forwarding) < 0) +		goto nla_put_failure; +#ifdef CONFIG_IPV6_MROUTE +	if ((type == -1 || type == NETCONFA_MC_FORWARDING) && +	    nla_put_s32(skb, NETCONFA_MC_FORWARDING, +			devconf->mc_forwarding) < 0) +		goto nla_put_failure; +#endif +	if ((type == -1 || type == NETCONFA_PROXY_NEIGH) && +	    nla_put_s32(skb, NETCONFA_PROXY_NEIGH, devconf->proxy_ndp) < 0) +		goto nla_put_failure; + +	return nlmsg_end(skb, nlh); + +nla_put_failure: +	nlmsg_cancel(skb, nlh); +	return -EMSGSIZE; +} + +void inet6_netconf_notify_devconf(struct net *net, int type, int ifindex, +				  struct ipv6_devconf *devconf) +{ +	struct sk_buff *skb; +	int err = -ENOBUFS; + +	skb = nlmsg_new(inet6_netconf_msgsize_devconf(type), GFP_ATOMIC); +	if (skb == NULL) +		goto errout; + +	err = inet6_netconf_fill_devconf(skb, ifindex, devconf, 0, 0, +					 RTM_NEWNETCONF, 0, type); +	if (err < 0) { +		/* -EMSGSIZE implies BUG in inet6_netconf_msgsize_devconf() */ +		WARN_ON(err == -EMSGSIZE); +		kfree_skb(skb); +		goto errout; +	} +	rtnl_notify(skb, net, 0, RTNLGRP_IPV6_NETCONF, NULL, GFP_ATOMIC); +	return; +errout: +	rtnl_set_sk_err(net, RTNLGRP_IPV6_NETCONF, err); +} + +static const struct nla_policy devconf_ipv6_policy[NETCONFA_MAX+1] = { +	[NETCONFA_IFINDEX]	= { .len = sizeof(int) }, +	[NETCONFA_FORWARDING]	= { .len = sizeof(int) }, +	[NETCONFA_PROXY_NEIGH]	= { .len = sizeof(int) }, +}; + +static int inet6_netconf_get_devconf(struct sk_buff *in_skb, +				     struct nlmsghdr *nlh) +{ +	struct net *net = sock_net(in_skb->sk); +	struct nlattr *tb[NETCONFA_MAX+1]; +	struct netconfmsg *ncm; +	struct sk_buff *skb; +	struct ipv6_devconf *devconf; +	struct inet6_dev *in6_dev; +	struct net_device *dev; +	int ifindex; +	int err; + +	err = nlmsg_parse(nlh, sizeof(*ncm), tb, NETCONFA_MAX, +			  devconf_ipv6_policy); +	if (err < 0) +		goto errout; + +	err = EINVAL; +	if (!tb[NETCONFA_IFINDEX]) +		goto errout; + +	ifindex = nla_get_s32(tb[NETCONFA_IFINDEX]); +	switch (ifindex) { +	case NETCONFA_IFINDEX_ALL: +		devconf = net->ipv6.devconf_all; +		break; +	case NETCONFA_IFINDEX_DEFAULT: +		devconf = net->ipv6.devconf_dflt; +		break; +	default: +		dev = __dev_get_by_index(net, ifindex); +		if (dev == NULL) +			goto errout; +		in6_dev = __in6_dev_get(dev); +		if (in6_dev == NULL) +			goto errout; +		devconf = &in6_dev->cnf; +		break; +	} + +	err = -ENOBUFS; +	skb = nlmsg_new(inet6_netconf_msgsize_devconf(-1), GFP_ATOMIC); +	if (skb == NULL) +		goto errout; + +	err = inet6_netconf_fill_devconf(skb, ifindex, devconf, +					 NETLINK_CB(in_skb).portid, +					 nlh->nlmsg_seq, RTM_NEWNETCONF, 0, +					 -1); +	if (err < 0) { +		/* -EMSGSIZE implies BUG in inet6_netconf_msgsize_devconf() */ +		WARN_ON(err == -EMSGSIZE); +		kfree_skb(skb); +		goto errout; +	} +	err = rtnl_unicast(skb, net, NETLINK_CB(in_skb).portid); +errout: +	return err; +} + +static int inet6_netconf_dump_devconf(struct sk_buff *skb, +				      struct netlink_callback *cb) +{ +	struct net *net = sock_net(skb->sk); +	int h, s_h; +	int idx, s_idx; +	struct net_device *dev; +	struct inet6_dev *idev; +	struct hlist_head *head; + +	s_h = cb->args[0]; +	s_idx = idx = cb->args[1]; + +	for (h = s_h; h < NETDEV_HASHENTRIES; h++, s_idx = 0) { +		idx = 0; +		head = &net->dev_index_head[h]; +		rcu_read_lock(); +		cb->seq = atomic_read(&net->ipv6.dev_addr_genid) ^ +			  net->dev_base_seq; +		hlist_for_each_entry_rcu(dev, head, index_hlist) { +			if (idx < s_idx) +				goto cont; +			idev = __in6_dev_get(dev); +			if (!idev) +				goto cont; + +			if (inet6_netconf_fill_devconf(skb, dev->ifindex, +						       &idev->cnf, +						       NETLINK_CB(cb->skb).portid, +						       cb->nlh->nlmsg_seq, +						       RTM_NEWNETCONF, +						       NLM_F_MULTI, +						       -1) <= 0) { +				rcu_read_unlock(); +				goto done; +			} +			nl_dump_check_consistent(cb, nlmsg_hdr(skb)); +cont: +			idx++; +		} +		rcu_read_unlock(); +	} +	if (h == NETDEV_HASHENTRIES) { +		if (inet6_netconf_fill_devconf(skb, NETCONFA_IFINDEX_ALL, +					       net->ipv6.devconf_all, +					       NETLINK_CB(cb->skb).portid, +					       cb->nlh->nlmsg_seq, +					       RTM_NEWNETCONF, NLM_F_MULTI, +					       -1) <= 0) +			goto done; +		else +			h++; +	} +	if (h == NETDEV_HASHENTRIES + 1) { +		if (inet6_netconf_fill_devconf(skb, NETCONFA_IFINDEX_DEFAULT, +					       net->ipv6.devconf_dflt, +					       NETLINK_CB(cb->skb).portid, +					       cb->nlh->nlmsg_seq, +					       RTM_NEWNETCONF, NLM_F_MULTI, +					       -1) <= 0) +			goto done; +		else +			h++; +	} +done: +	cb->args[0] = h; +	cb->args[1] = idx; + +	return skb->len; +} +  #ifdef CONFIG_SYSCTL  static void dev_forward_change(struct inet6_dev *idev)  { @@ -474,11 +658,16 @@ static void dev_forward_change(struct inet6_dev *idev)  	dev = idev->dev;  	if (idev->cnf.forwarding)  		dev_disable_lro(dev); -	if (dev && (dev->flags & IFF_MULTICAST)) { -		if (idev->cnf.forwarding) +	if (dev->flags & IFF_MULTICAST) { +		if (idev->cnf.forwarding) {  			ipv6_dev_mc_inc(dev, &in6addr_linklocal_allrouters); -		else +			ipv6_dev_mc_inc(dev, &in6addr_interfacelocal_allrouters); +			ipv6_dev_mc_inc(dev, &in6addr_sitelocal_allrouters); +		} else {  			ipv6_dev_mc_dec(dev, &in6addr_linklocal_allrouters); +			ipv6_dev_mc_dec(dev, &in6addr_interfacelocal_allrouters); +			ipv6_dev_mc_dec(dev, &in6addr_sitelocal_allrouters); +		}  	}  	list_for_each_entry(ifa, &idev->addr_list, if_list) { @@ -489,6 +678,8 @@ static void dev_forward_change(struct inet6_dev *idev)  		else  			addrconf_leave_anycast(ifa);  	} +	inet6_netconf_notify_devconf(dev_net(dev), NETCONFA_FORWARDING, +				     dev->ifindex, &idev->cnf);  } @@ -497,8 +688,7 @@ static void addrconf_forward_change(struct net *net, __s32 newf)  	struct net_device *dev;  	struct inet6_dev *idev; -	rcu_read_lock(); -	for_each_netdev_rcu(net, dev) { +	for_each_netdev(net, dev) {  		idev = __in6_dev_get(dev);  		if (idev) {  			int changed = (!idev->cnf.forwarding) ^ (!newf); @@ -507,64 +697,68 @@ static void addrconf_forward_change(struct net *net, __s32 newf)  				dev_forward_change(idev);  		}  	} -	rcu_read_unlock();  } -static int addrconf_fixup_forwarding(struct ctl_table *table, int *p, int old) +static int addrconf_fixup_forwarding(struct ctl_table *table, int *p, int newf)  {  	struct net *net; +	int old; + +	if (!rtnl_trylock()) +		return restart_syscall();  	net = (struct net *)table->extra2; -	if (p == &net->ipv6.devconf_dflt->forwarding) +	old = *p; +	*p = newf; + +	if (p == &net->ipv6.devconf_dflt->forwarding) { +		if ((!newf) ^ (!old)) +			inet6_netconf_notify_devconf(net, NETCONFA_FORWARDING, +						     NETCONFA_IFINDEX_DEFAULT, +						     net->ipv6.devconf_dflt); +		rtnl_unlock();  		return 0; - -	if (!rtnl_trylock()) { -		/* Restore the original values before restarting */ -		*p = old; -		return restart_syscall();  	}  	if (p == &net->ipv6.devconf_all->forwarding) { -		__s32 newf = net->ipv6.devconf_all->forwarding;  		net->ipv6.devconf_dflt->forwarding = newf;  		addrconf_forward_change(net, newf); -	} else if ((!*p) ^ (!old)) +		if ((!newf) ^ (!old)) +			inet6_netconf_notify_devconf(net, NETCONFA_FORWARDING, +						     NETCONFA_IFINDEX_ALL, +						     net->ipv6.devconf_all); +	} else if ((!newf) ^ (!old))  		dev_forward_change((struct inet6_dev *)table->extra1);  	rtnl_unlock(); -	if (*p) +	if (newf)  		rt6_purge_dflt_routers(net);  	return 1;  }  #endif -static void inet6_ifa_finish_destroy_rcu(struct rcu_head *head) -{ -	struct inet6_ifaddr *ifp = container_of(head, struct inet6_ifaddr, rcu); -	kfree(ifp); -} -  /* Nobody refers to this ifaddr, destroy it */  void inet6_ifa_finish_destroy(struct inet6_ifaddr *ifp)  {  	WARN_ON(!hlist_unhashed(&ifp->addr_lst));  #ifdef NET_REFCNT_DEBUG -	printk(KERN_DEBUG "inet6_ifa_finish_destroy\n"); +	pr_debug("%s\n", __func__);  #endif  	in6_dev_put(ifp->idev); -	if (del_timer(&ifp->timer)) -		pr_notice("Timer is still running, when freeing ifa=%p\n", ifp); +	if (cancel_delayed_work(&ifp->dad_work)) +		pr_notice("delayed DAD work was pending while freeing ifa=%p\n", +			  ifp);  	if (ifp->state != INET6_IFADDR_STATE_DEAD) { -		pr_warning("Freeing alive inet6 address %p\n", ifp); +		pr_warn("Freeing alive inet6 address %p\n", ifp);  		return;  	} -	dst_release(&ifp->rt->dst); +	ip6_rt_put(ifp->rt); -	call_rcu(&ifp->rcu, inet6_ifa_finish_destroy_rcu); +	kfree_rcu(ifp, rcu);  }  static void @@ -587,22 +781,17 @@ ipv6_link_dev_addr(struct inet6_dev *idev, struct inet6_ifaddr *ifp)  	list_add_tail(&ifp->if_list, p);  } -static u32 ipv6_addr_hash(const struct in6_addr *addr) +static u32 inet6_addr_hash(const struct in6_addr *addr)  { -	/* -	 * We perform the hash function over the last 64 bits of the address -	 * This will include the IEEE address token on links that support it. -	 */ -	return jhash_2words((__force u32)addr->s6_addr32[2], -			    (__force u32)addr->s6_addr32[3], 0) -		& (IN6_ADDR_HSIZE - 1); +	return hash_32(ipv6_addr_hash(addr), IN6_ADDR_HSIZE_SHIFT);  }  /* On success it returns ifp with increased reference count */  static struct inet6_ifaddr * -ipv6_add_addr(struct inet6_dev *idev, const struct in6_addr *addr, int pfxlen, -	      int scope, u32 flags) +ipv6_add_addr(struct inet6_dev *idev, const struct in6_addr *addr, +	      const struct in6_addr *peer_addr, int pfxlen, +	      int scope, u32 flags, u32 valid_lft, u32 prefered_lft)  {  	struct inet6_ifaddr *ifa = NULL;  	struct rt6_info *rt; @@ -631,7 +820,7 @@ ipv6_add_addr(struct inet6_dev *idev, const struct in6_addr *addr, int pfxlen,  	/* Ignore adding duplicate addresses on an interface */  	if (ipv6_chk_same_addr(dev_net(idev->dev), addr, idev->dev)) { -		ADBG(("ipv6_add_addr: already assigned\n")); +		ADBG("ipv6_add_addr: already assigned\n");  		err = -EEXIST;  		goto out;  	} @@ -639,48 +828,44 @@ ipv6_add_addr(struct inet6_dev *idev, const struct in6_addr *addr, int pfxlen,  	ifa = kzalloc(sizeof(struct inet6_ifaddr), GFP_ATOMIC);  	if (ifa == NULL) { -		ADBG(("ipv6_add_addr: malloc failed\n")); +		ADBG("ipv6_add_addr: malloc failed\n");  		err = -ENOBUFS;  		goto out;  	} -	rt = addrconf_dst_alloc(idev, addr, 0); +	rt = addrconf_dst_alloc(idev, addr, false);  	if (IS_ERR(rt)) {  		err = PTR_ERR(rt);  		goto out;  	} -	ipv6_addr_copy(&ifa->addr, addr); +	neigh_parms_data_state_setall(idev->nd_parms); + +	ifa->addr = *addr; +	if (peer_addr) +		ifa->peer_addr = *peer_addr;  	spin_lock_init(&ifa->lock);  	spin_lock_init(&ifa->state_lock); -	init_timer(&ifa->timer); +	INIT_DELAYED_WORK(&ifa->dad_work, addrconf_dad_work);  	INIT_HLIST_NODE(&ifa->addr_lst); -	ifa->timer.data = (unsigned long) ifa;  	ifa->scope = scope;  	ifa->prefix_len = pfxlen;  	ifa->flags = flags | IFA_F_TENTATIVE; +	ifa->valid_lft = valid_lft; +	ifa->prefered_lft = prefered_lft;  	ifa->cstamp = ifa->tstamp = jiffies; +	ifa->tokenized = false;  	ifa->rt = rt; -	/* -	 * part one of RFC 4429, section 3.3 -	 * We should not configure an address as -	 * optimistic if we do not yet know the link -	 * layer address of our nexhop router -	 */ - -	if (rt->rt6i_nexthop == NULL) -		ifa->flags &= ~IFA_F_OPTIMISTIC; -  	ifa->idev = idev;  	in6_dev_hold(idev);  	/* For caller */  	in6_ifa_hold(ifa);  	/* Add to big hash table */ -	hash = ipv6_addr_hash(addr); +	hash = inet6_addr_hash(addr);  	hlist_add_head_rcu(&ifa->addr_lst, &inet6_addr_lst[hash]);  	spin_unlock(&addrconf_hash_lock); @@ -689,12 +874,10 @@ ipv6_add_addr(struct inet6_dev *idev, const struct in6_addr *addr, int pfxlen,  	/* Add to inet6_dev unicast addr list. */  	ipv6_link_dev_addr(idev, ifa); -#ifdef CONFIG_IPV6_PRIVACY  	if (ifa->flags&IFA_F_TEMPORARY) {  		list_add(&ifa->tmp_list, &idev->tempaddr_list);  		in6_ifa_hold(ifa);  	} -#endif  	in6_ifa_hold(ifa);  	write_unlock(&idev->lock); @@ -702,7 +885,7 @@ out2:  	rcu_read_unlock_bh();  	if (likely(err == 0)) -		atomic_notifier_call_chain(&inet6addr_chain, NETDEV_UP, ifa); +		inet6addr_notifier_call_chain(NETDEV_UP, ifa);  	else {  		kfree(ifa);  		ifa = ERR_PTR(err); @@ -714,18 +897,97 @@ out:  	goto out2;  } +enum cleanup_prefix_rt_t { +	CLEANUP_PREFIX_RT_NOP,    /* no cleanup action for prefix route */ +	CLEANUP_PREFIX_RT_DEL,    /* delete the prefix route */ +	CLEANUP_PREFIX_RT_EXPIRE, /* update the lifetime of the prefix route */ +}; + +/* + * Check, whether the prefix for ifp would still need a prefix route + * after deleting ifp. The function returns one of the CLEANUP_PREFIX_RT_* + * constants. + * + * 1) we don't purge prefix if address was not permanent. + *    prefix is managed by its own lifetime. + * 2) we also don't purge, if the address was IFA_F_NOPREFIXROUTE. + * 3) if there are no addresses, delete prefix. + * 4) if there are still other permanent address(es), + *    corresponding prefix is still permanent. + * 5) if there are still other addresses with IFA_F_NOPREFIXROUTE, + *    don't purge the prefix, assume user space is managing it. + * 6) otherwise, update prefix lifetime to the + *    longest valid lifetime among the corresponding + *    addresses on the device. + *    Note: subsequent RA will update lifetime. + **/ +static enum cleanup_prefix_rt_t +check_cleanup_prefix_route(struct inet6_ifaddr *ifp, unsigned long *expires) +{ +	struct inet6_ifaddr *ifa; +	struct inet6_dev *idev = ifp->idev; +	unsigned long lifetime; +	enum cleanup_prefix_rt_t action = CLEANUP_PREFIX_RT_DEL; + +	*expires = jiffies; + +	list_for_each_entry(ifa, &idev->addr_list, if_list) { +		if (ifa == ifp) +			continue; +		if (!ipv6_prefix_equal(&ifa->addr, &ifp->addr, +				       ifp->prefix_len)) +			continue; +		if (ifa->flags & (IFA_F_PERMANENT | IFA_F_NOPREFIXROUTE)) +			return CLEANUP_PREFIX_RT_NOP; + +		action = CLEANUP_PREFIX_RT_EXPIRE; + +		spin_lock(&ifa->lock); + +		lifetime = addrconf_timeout_fixup(ifa->valid_lft, HZ); +		/* +		 * Note: Because this address is +		 * not permanent, lifetime < +		 * LONG_MAX / HZ here. +		 */ +		if (time_before(*expires, ifa->tstamp + lifetime * HZ)) +			*expires = ifa->tstamp + lifetime * HZ; +		spin_unlock(&ifa->lock); +	} + +	return action; +} + +static void +cleanup_prefix_route(struct inet6_ifaddr *ifp, unsigned long expires, bool del_rt) +{ +	struct rt6_info *rt; + +	rt = addrconf_get_prefix_route(&ifp->addr, +				       ifp->prefix_len, +				       ifp->idev->dev, +				       0, RTF_GATEWAY | RTF_DEFAULT); +	if (rt) { +		if (del_rt) +			ip6_del_rt(rt); +		else { +			if (!(rt->rt6i_flags & RTF_EXPIRES)) +				rt6_set_expires(rt, expires); +			ip6_rt_put(rt); +		} +	} +} + +  /* This function wants to get referenced ifp and releases it before return */  static void ipv6_del_addr(struct inet6_ifaddr *ifp)  { -	struct inet6_ifaddr *ifa, *ifn; -	struct inet6_dev *idev = ifp->idev;  	int state; -	int hash; -	int deleted = 0, onlink = 0; -	unsigned long expires = jiffies; +	enum cleanup_prefix_rt_t action = CLEANUP_PREFIX_RT_NOP; +	unsigned long expires; -	hash = ipv6_addr_hash(&ifp->addr); +	ASSERT_RTNL();  	spin_lock_bh(&ifp->state_lock);  	state = ifp->state; @@ -739,8 +1001,8 @@ static void ipv6_del_addr(struct inet6_ifaddr *ifp)  	hlist_del_init_rcu(&ifp->addr_lst);  	spin_unlock_bh(&addrconf_hash_lock); -	write_lock_bh(&idev->lock); -#ifdef CONFIG_IPV6_PRIVACY +	write_lock_bh(&ifp->idev->lock); +  	if (ifp->flags&IFA_F_TEMPORARY) {  		list_del(&ifp->tmp_list);  		if (ifp->ifpub) { @@ -749,105 +1011,44 @@ static void ipv6_del_addr(struct inet6_ifaddr *ifp)  		}  		__in6_ifa_put(ifp);  	} -#endif -	list_for_each_entry_safe(ifa, ifn, &idev->addr_list, if_list) { -		if (ifa == ifp) { -			list_del_init(&ifp->if_list); -			__in6_ifa_put(ifp); +	if (ifp->flags & IFA_F_PERMANENT && !(ifp->flags & IFA_F_NOPREFIXROUTE)) +		action = check_cleanup_prefix_route(ifp, &expires); -			if (!(ifp->flags & IFA_F_PERMANENT) || onlink > 0) -				break; -			deleted = 1; -			continue; -		} else if (ifp->flags & IFA_F_PERMANENT) { -			if (ipv6_prefix_equal(&ifa->addr, &ifp->addr, -					      ifp->prefix_len)) { -				if (ifa->flags & IFA_F_PERMANENT) { -					onlink = 1; -					if (deleted) -						break; -				} else { -					unsigned long lifetime; - -					if (!onlink) -						onlink = -1; - -					spin_lock(&ifa->lock); - -					lifetime = addrconf_timeout_fixup(ifa->valid_lft, HZ); -					/* -					 * Note: Because this address is -					 * not permanent, lifetime < -					 * LONG_MAX / HZ here. -					 */ -					if (time_before(expires, -							ifa->tstamp + lifetime * HZ)) -						expires = ifa->tstamp + lifetime * HZ; -					spin_unlock(&ifa->lock); -				} -			} -		} -	} -	write_unlock_bh(&idev->lock); +	list_del_init(&ifp->if_list); +	__in6_ifa_put(ifp); -	addrconf_del_timer(ifp); +	write_unlock_bh(&ifp->idev->lock); -	ipv6_ifa_notify(RTM_DELADDR, ifp); +	addrconf_del_dad_work(ifp); -	atomic_notifier_call_chain(&inet6addr_chain, NETDEV_DOWN, ifp); +	ipv6_ifa_notify(RTM_DELADDR, ifp); -	/* -	 * Purge or update corresponding prefix -	 * -	 * 1) we don't purge prefix here if address was not permanent. -	 *    prefix is managed by its own lifetime. -	 * 2) if there're no addresses, delete prefix. -	 * 3) if there're still other permanent address(es), -	 *    corresponding prefix is still permanent. -	 * 4) otherwise, update prefix lifetime to the -	 *    longest valid lifetime among the corresponding -	 *    addresses on the device. -	 *    Note: subsequent RA will update lifetime. -	 * -	 * --yoshfuji -	 */ -	if ((ifp->flags & IFA_F_PERMANENT) && onlink < 1) { -		struct in6_addr prefix; -		struct rt6_info *rt; -		struct net *net = dev_net(ifp->idev->dev); -		ipv6_addr_prefix(&prefix, &ifp->addr, ifp->prefix_len); -		rt = rt6_lookup(net, &prefix, NULL, ifp->idev->dev->ifindex, 1); +	inet6addr_notifier_call_chain(NETDEV_DOWN, ifp); -		if (rt && addrconf_is_prefix_route(rt)) { -			if (onlink == 0) { -				ip6_del_rt(rt); -				rt = NULL; -			} else if (!(rt->rt6i_flags & RTF_EXPIRES)) { -				rt->rt6i_expires = expires; -				rt->rt6i_flags |= RTF_EXPIRES; -			} -		} -		dst_release(&rt->dst); +	if (action != CLEANUP_PREFIX_RT_NOP) { +		cleanup_prefix_route(ifp, expires, +			action == CLEANUP_PREFIX_RT_DEL);  	} +	/* clean up prefsrc entries */ +	rt6_remove_prefsrc(ifp);  out:  	in6_ifa_put(ifp);  } -#ifdef CONFIG_IPV6_PRIVACY  static int ipv6_create_tempaddr(struct inet6_ifaddr *ifp, struct inet6_ifaddr *ift)  {  	struct inet6_dev *idev = ifp->idev;  	struct in6_addr addr, *tmpaddr; -	unsigned long tmp_prefered_lft, tmp_valid_lft, tmp_cstamp, tmp_tstamp, age; +	unsigned long tmp_prefered_lft, tmp_valid_lft, tmp_tstamp, age;  	unsigned long regen_advance;  	int tmp_plen;  	int ret = 0; -	int max_addresses;  	u32 addr_flags; +	unsigned long now = jiffies; -	write_lock(&idev->lock); +	write_lock_bh(&idev->lock);  	if (ift) {  		spin_lock_bh(&ift->lock);  		memcpy(&addr.s6_addr[8], &ift->addr.s6_addr[8], 8); @@ -859,9 +1060,8 @@ static int ipv6_create_tempaddr(struct inet6_ifaddr *ifp, struct inet6_ifaddr *i  retry:  	in6_dev_hold(idev);  	if (idev->cnf.use_tempaddr <= 0) { -		write_unlock(&idev->lock); -		printk(KERN_INFO -			"ipv6_create_tempaddr(): use_tempaddr is disabled.\n"); +		write_unlock_bh(&idev->lock); +		pr_info("%s: use_tempaddr is disabled\n", __func__);  		in6_dev_put(idev);  		ret = -1;  		goto out; @@ -870,27 +1070,18 @@ retry:  	if (ifp->regen_count++ >= idev->cnf.regen_max_retry) {  		idev->cnf.use_tempaddr = -1;	/*XXX*/  		spin_unlock_bh(&ifp->lock); -		write_unlock(&idev->lock); -		printk(KERN_WARNING -			"ipv6_create_tempaddr(): regeneration time exceeded. disabled temporary address support.\n"); +		write_unlock_bh(&idev->lock); +		pr_warn("%s: regeneration time exceeded - disabled temporary address support\n", +			__func__);  		in6_dev_put(idev);  		ret = -1;  		goto out;  	}  	in6_ifa_hold(ifp);  	memcpy(addr.s6_addr, ifp->addr.s6_addr, 8); -	if (__ipv6_try_regen_rndid(idev, tmpaddr) < 0) { -		spin_unlock_bh(&ifp->lock); -		write_unlock(&idev->lock); -		printk(KERN_WARNING -			"ipv6_create_tempaddr(): regeneration of randomized interface id failed.\n"); -		in6_ifa_put(ifp); -		in6_dev_put(idev); -		ret = -1; -		goto out; -	} +	__ipv6_try_regen_rndid(idev, tmpaddr);  	memcpy(&addr.s6_addr[8], idev->rndid, 8); -	age = (jiffies - ifp->tstamp) / HZ; +	age = (now - ifp->tstamp) / HZ;  	tmp_valid_lft = min_t(__u32,  			      ifp->valid_lft,  			      idev->cnf.temp_valid_lft + age); @@ -899,22 +1090,23 @@ retry:  				 idev->cnf.temp_prefered_lft + age -  				 idev->cnf.max_desync_factor);  	tmp_plen = ifp->prefix_len; -	max_addresses = idev->cnf.max_addresses; -	tmp_cstamp = ifp->cstamp;  	tmp_tstamp = ifp->tstamp;  	spin_unlock_bh(&ifp->lock);  	regen_advance = idev->cnf.regen_max_retry *  	                idev->cnf.dad_transmits * -	                idev->nd_parms->retrans_time / HZ; -	write_unlock(&idev->lock); +	                NEIGH_VAR(idev->nd_parms, RETRANS_TIME) / HZ; +	write_unlock_bh(&idev->lock);  	/* A temporary address is created only if this calculated Preferred  	 * Lifetime is greater than REGEN_ADVANCE time units.  In particular,  	 * an implementation must not create a temporary address with a zero  	 * Preferred Lifetime. +	 * Use age calculation as in addrconf_verify to avoid unnecessary +	 * temporary addresses being generated.  	 */ -	if (tmp_prefered_lft <= regen_advance) { +	age = (now - tmp_tstamp + ADDRCONF_TIMER_FUZZ_MINUS) / HZ; +	if (tmp_prefered_lft <= regen_advance + age) {  		in6_ifa_put(ifp);  		in6_dev_put(idev);  		ret = -1; @@ -926,36 +1118,30 @@ retry:  	if (ifp->flags & IFA_F_OPTIMISTIC)  		addr_flags |= IFA_F_OPTIMISTIC; -	ift = !max_addresses || -	      ipv6_count_addresses(idev) < max_addresses ? -		ipv6_add_addr(idev, &addr, tmp_plen, -			      ipv6_addr_type(&addr)&IPV6_ADDR_SCOPE_MASK, -			      addr_flags) : NULL; -	if (!ift || IS_ERR(ift)) { +	ift = ipv6_add_addr(idev, &addr, NULL, tmp_plen, +			    ipv6_addr_scope(&addr), addr_flags, +			    tmp_valid_lft, tmp_prefered_lft); +	if (IS_ERR(ift)) {  		in6_ifa_put(ifp);  		in6_dev_put(idev); -		printk(KERN_INFO -			"ipv6_create_tempaddr(): retry temporary address regeneration.\n"); +		pr_info("%s: retry temporary address regeneration\n", __func__);  		tmpaddr = &addr; -		write_lock(&idev->lock); +		write_lock_bh(&idev->lock);  		goto retry;  	}  	spin_lock_bh(&ift->lock);  	ift->ifpub = ifp; -	ift->valid_lft = tmp_valid_lft; -	ift->prefered_lft = tmp_prefered_lft; -	ift->cstamp = tmp_cstamp; +	ift->cstamp = now;  	ift->tstamp = tmp_tstamp;  	spin_unlock_bh(&ift->lock); -	addrconf_dad_start(ift, 0); +	addrconf_dad_start(ift);  	in6_ifa_put(ift);  	in6_dev_put(idev);  out:  	return ret;  } -#endif  /*   *	Choose an appropriate source address (RFC3484) @@ -970,9 +1156,7 @@ enum {  #endif  	IPV6_SADDR_RULE_OIF,  	IPV6_SADDR_RULE_LABEL, -#ifdef CONFIG_IPV6_PRIVACY  	IPV6_SADDR_RULE_PRIVACY, -#endif  	IPV6_SADDR_RULE_ORCHID,  	IPV6_SADDR_RULE_PREFIX,  	IPV6_SADDR_RULE_MAX @@ -1043,7 +1227,7 @@ static int ipv6_get_saddr_eval(struct net *net,  		 *       |             d is scope of the destination.  		 *  B-d  |  \  		 *       |   \      <- smaller scope is better if -		 *  B-15 |    \        if scope is enough for destinaion. +		 *  B-15 |    \        if scope is enough for destination.  		 *       |             ret = B - scope (-1 <= scope >= d <= 15).  		 * d-C-1 | /  		 *       |/         <- greater is better @@ -1086,11 +1270,10 @@ static int ipv6_get_saddr_eval(struct net *net,  				      &score->ifa->addr, score->addr_type,  				      score->ifa->idev->dev->ifindex) == dst->label;  		break; -#ifdef CONFIG_IPV6_PRIVACY  	case IPV6_SADDR_RULE_PRIVACY:  	    {  		/* Rule 7: Prefer public address -		 * Note: prefer temprary address if use_tempaddr >= 2 +		 * Note: prefer temporary address if use_tempaddr >= 2  		 */  		int preftmp = dst->prefs & (IPV6_PREFER_SRC_PUBLIC|IPV6_PREFER_SRC_TMP) ?  				!!(dst->prefs & IPV6_PREFER_SRC_TMP) : @@ -1098,7 +1281,6 @@ static int ipv6_get_saddr_eval(struct net *net,  		ret = (!(score->ifa->flags & IFA_F_TEMPORARY)) ^ preftmp;  		break;  	    } -#endif  	case IPV6_SADDR_RULE_ORCHID:  		/* Rule 8-: Prefer ORCHID vs ORCHID or  		 *	    non-ORCHID vs non-ORCHID @@ -1108,8 +1290,10 @@ static int ipv6_get_saddr_eval(struct net *net,  		break;  	case IPV6_SADDR_RULE_PREFIX:  		/* Rule 8: Use longest matching prefix */ -		score->matchlen = ret = ipv6_addr_diff(&score->ifa->addr, -						       dst->addr); +		ret = ipv6_addr_diff(&score->ifa->addr, dst->addr); +		if (ret > score->ifa->prefix_len) +			ret = score->ifa->prefix_len; +		score->matchlen = ret;  		break;  	default:  		ret = 0; @@ -1122,7 +1306,7 @@ out:  	return ret;  } -int ipv6_dev_get_saddr(struct net *net, struct net_device *dst_dev, +int ipv6_dev_get_saddr(struct net *net, const struct net_device *dst_dev,  		       const struct in6_addr *daddr, unsigned int prefs,  		       struct in6_addr *saddr)  { @@ -1243,14 +1427,33 @@ try_nextdev:  	if (!hiscore->ifa)  		return -EADDRNOTAVAIL; -	ipv6_addr_copy(saddr, &hiscore->ifa->addr); +	*saddr = hiscore->ifa->addr;  	in6_ifa_put(hiscore->ifa);  	return 0;  }  EXPORT_SYMBOL(ipv6_dev_get_saddr); +int __ipv6_get_lladdr(struct inet6_dev *idev, struct in6_addr *addr, +		      u32 banned_flags) +{ +	struct inet6_ifaddr *ifp; +	int err = -EADDRNOTAVAIL; + +	list_for_each_entry_reverse(ifp, &idev->addr_list, if_list) { +		if (ifp->scope > IFA_LINK) +			break; +		if (ifp->scope == IFA_LINK && +		    !(ifp->flags & banned_flags)) { +			*addr = ifp->addr; +			err = 0; +			break; +		} +	} +	return err; +} +  int ipv6_get_lladdr(struct net_device *dev, struct in6_addr *addr, -		    unsigned char banned_flags) +		    u32 banned_flags)  {  	struct inet6_dev *idev;  	int err = -EADDRNOTAVAIL; @@ -1258,17 +1461,8 @@ int ipv6_get_lladdr(struct net_device *dev, struct in6_addr *addr,  	rcu_read_lock();  	idev = __in6_dev_get(dev);  	if (idev) { -		struct inet6_ifaddr *ifp; -  		read_lock_bh(&idev->lock); -		list_for_each_entry(ifp, &idev->addr_list, if_list) { -			if (ifp->scope == IFA_LINK && -			    !(ifp->flags & banned_flags)) { -				ipv6_addr_copy(addr, &ifp->addr); -				err = 0; -				break; -			} -		} +		err = __ipv6_get_lladdr(idev, addr, banned_flags);  		read_unlock_bh(&idev->lock);  	}  	rcu_read_unlock(); @@ -1287,15 +1481,14 @@ static int ipv6_count_addresses(struct inet6_dev *idev)  	return cnt;  } -int ipv6_chk_addr(struct net *net, struct in6_addr *addr, -		  struct net_device *dev, int strict) +int ipv6_chk_addr(struct net *net, const struct in6_addr *addr, +		  const struct net_device *dev, int strict)  {  	struct inet6_ifaddr *ifp; -	struct hlist_node *node; -	unsigned int hash = ipv6_addr_hash(addr); +	unsigned int hash = inet6_addr_hash(addr);  	rcu_read_lock_bh(); -	hlist_for_each_entry_rcu(ifp, node, &inet6_addr_lst[hash], addr_lst) { +	hlist_for_each_entry_rcu(ifp, &inet6_addr_lst[hash], addr_lst) {  		if (!net_eq(dev_net(ifp->idev->dev), net))  			continue;  		if (ipv6_addr_equal(&ifp->addr, addr) && @@ -1315,11 +1508,10 @@ EXPORT_SYMBOL(ipv6_chk_addr);  static bool ipv6_chk_same_addr(struct net *net, const struct in6_addr *addr,  			       struct net_device *dev)  { -	unsigned int hash = ipv6_addr_hash(addr); +	unsigned int hash = inet6_addr_hash(addr);  	struct inet6_ifaddr *ifp; -	struct hlist_node *node; -	hlist_for_each_entry(ifp, node, &inet6_addr_lst[hash], addr_lst) { +	hlist_for_each_entry(ifp, &inet6_addr_lst[hash], addr_lst) {  		if (!net_eq(dev_net(ifp->idev->dev), net))  			continue;  		if (ipv6_addr_equal(&ifp->addr, addr)) { @@ -1330,7 +1522,34 @@ static bool ipv6_chk_same_addr(struct net *net, const struct in6_addr *addr,  	return false;  } -int ipv6_chk_prefix(struct in6_addr *addr, struct net_device *dev) +/* Compares an address/prefix_len with addresses on device @dev. + * If one is found it returns true. + */ +bool ipv6_chk_custom_prefix(const struct in6_addr *addr, +	const unsigned int prefix_len, struct net_device *dev) +{ +	struct inet6_dev *idev; +	struct inet6_ifaddr *ifa; +	bool ret = false; + +	rcu_read_lock(); +	idev = __in6_dev_get(dev); +	if (idev) { +		read_lock_bh(&idev->lock); +		list_for_each_entry(ifa, &idev->addr_list, if_list) { +			ret = ipv6_prefix_equal(addr, &ifa->addr, prefix_len); +			if (ret) +				break; +		} +		read_unlock_bh(&idev->lock); +	} +	rcu_read_unlock(); + +	return ret; +} +EXPORT_SYMBOL(ipv6_chk_custom_prefix); + +int ipv6_chk_prefix(const struct in6_addr *addr, struct net_device *dev)  {  	struct inet6_dev *idev;  	struct inet6_ifaddr *ifa; @@ -1352,18 +1571,16 @@ int ipv6_chk_prefix(struct in6_addr *addr, struct net_device *dev)  	rcu_read_unlock();  	return onlink;  } -  EXPORT_SYMBOL(ipv6_chk_prefix);  struct inet6_ifaddr *ipv6_get_ifaddr(struct net *net, const struct in6_addr *addr,  				     struct net_device *dev, int strict)  {  	struct inet6_ifaddr *ifp, *result = NULL; -	unsigned int hash = ipv6_addr_hash(addr); -	struct hlist_node *node; +	unsigned int hash = inet6_addr_hash(addr);  	rcu_read_lock_bh(); -	hlist_for_each_entry_rcu_bh(ifp, node, &inet6_addr_lst[hash], addr_lst) { +	hlist_for_each_entry_rcu_bh(ifp, &inet6_addr_lst[hash], addr_lst) {  		if (!net_eq(dev_net(ifp->idev->dev), net))  			continue;  		if (ipv6_addr_equal(&ifp->addr, addr)) { @@ -1386,7 +1603,7 @@ static void addrconf_dad_stop(struct inet6_ifaddr *ifp, int dad_failed)  {  	if (ifp->flags&IFA_F_PERMANENT) {  		spin_lock_bh(&ifp->lock); -		addrconf_del_timer(ifp); +		addrconf_del_dad_work(ifp);  		ifp->flags |= IFA_F_TENTATIVE;  		if (dad_failed)  			ifp->flags |= IFA_F_DADFAILED; @@ -1394,7 +1611,6 @@ static void addrconf_dad_stop(struct inet6_ifaddr *ifp, int dad_failed)  		if (dad_failed)  			ipv6_ifa_notify(0, ifp);  		in6_ifa_put(ifp); -#ifdef CONFIG_IPV6_PRIVACY  	} else if (ifp->flags&IFA_F_TEMPORARY) {  		struct inet6_ifaddr *ifpub;  		spin_lock_bh(&ifp->lock); @@ -1408,21 +1624,21 @@ static void addrconf_dad_stop(struct inet6_ifaddr *ifp, int dad_failed)  			spin_unlock_bh(&ifp->lock);  		}  		ipv6_del_addr(ifp); -#endif -	} else +	} else {  		ipv6_del_addr(ifp); +	}  }  static int addrconf_dad_end(struct inet6_ifaddr *ifp)  {  	int err = -ENOENT; -	spin_lock(&ifp->state_lock); +	spin_lock_bh(&ifp->state_lock);  	if (ifp->state == INET6_IFADDR_STATE_DAD) {  		ifp->state = INET6_IFADDR_STATE_POSTDAD;  		err = 0;  	} -	spin_unlock(&ifp->state_lock); +	spin_unlock_bh(&ifp->state_lock);  	return err;  } @@ -1436,9 +1652,8 @@ void addrconf_dad_failure(struct inet6_ifaddr *ifp)  		return;  	} -	if (net_ratelimit()) -		printk(KERN_INFO "%s: IPv6 duplicate address %pI6c detected!\n", -			ifp->idev->dev->name, &ifp->addr); +	net_info_ratelimited("%s: IPv6 duplicate address %pI6c detected!\n", +			     ifp->idev->dev->name, &ifp->addr);  	if (idev->cnf.accept_dad > 1 && !idev->cnf.disable_ipv6) {  		struct in6_addr addr; @@ -1451,20 +1666,27 @@ void addrconf_dad_failure(struct inet6_ifaddr *ifp)  			/* DAD failed for link-local based on MAC address */  			idev->cnf.disable_ipv6 = 1; -			printk(KERN_INFO "%s: IPv6 being disabled!\n", +			pr_info("%s: IPv6 being disabled!\n",  				ifp->idev->dev->name);  		}  	} -	addrconf_dad_stop(ifp, 1); +	spin_lock_bh(&ifp->state_lock); +	/* transition from _POSTDAD to _ERRDAD */ +	ifp->state = INET6_IFADDR_STATE_ERRDAD; +	spin_unlock_bh(&ifp->state_lock); + +	addrconf_mod_dad_work(ifp, 0);  }  /* Join to solicited addr multicast group. */ -void addrconf_join_solict(struct net_device *dev, struct in6_addr *addr) +void addrconf_join_solict(struct net_device *dev, const struct in6_addr *addr)  {  	struct in6_addr maddr; +	ASSERT_RTNL(); +  	if (dev->flags&(IFF_LOOPBACK|IFF_NOARP))  		return; @@ -1472,10 +1694,12 @@ void addrconf_join_solict(struct net_device *dev, struct in6_addr *addr)  	ipv6_dev_mc_inc(dev, &maddr);  } -void addrconf_leave_solict(struct inet6_dev *idev, struct in6_addr *addr) +void addrconf_leave_solict(struct inet6_dev *idev, const struct in6_addr *addr)  {  	struct in6_addr maddr; +	ASSERT_RTNL(); +  	if (idev->dev->flags&(IFF_LOOPBACK|IFF_NOARP))  		return; @@ -1486,6 +1710,11 @@ void addrconf_leave_solict(struct inet6_dev *idev, struct in6_addr *addr)  static void addrconf_join_anycast(struct inet6_ifaddr *ifp)  {  	struct in6_addr addr; + +	ASSERT_RTNL(); + +	if (ifp->prefix_len >= 127) /* RFC 6164 */ +		return;  	ipv6_addr_prefix(&addr, &ifp->addr, ifp->prefix_len);  	if (ipv6_addr_any(&addr))  		return; @@ -1495,6 +1724,11 @@ static void addrconf_join_anycast(struct inet6_ifaddr *ifp)  static void addrconf_leave_anycast(struct inet6_ifaddr *ifp)  {  	struct in6_addr addr; + +	ASSERT_RTNL(); + +	if (ifp->prefix_len >= 127) /* RFC 6164 */ +		return;  	ipv6_addr_prefix(&addr, &ifp->addr, ifp->prefix_len);  	if (ipv6_addr_any(&addr))  		return; @@ -1532,13 +1766,36 @@ static int addrconf_ifid_eui48(u8 *eui, struct net_device *dev)  	return 0;  } +static int addrconf_ifid_eui64(u8 *eui, struct net_device *dev) +{ +	if (dev->addr_len != IEEE802154_ADDR_LEN) +		return -1; +	memcpy(eui, dev->dev_addr, 8); +	eui[0] ^= 2; +	return 0; +} + +static int addrconf_ifid_ieee1394(u8 *eui, struct net_device *dev) +{ +	union fwnet_hwaddr *ha; + +	if (dev->addr_len != FWNET_ALEN) +		return -1; + +	ha = (union fwnet_hwaddr *)dev->dev_addr; + +	memcpy(eui, &ha->uc.uniq_id, sizeof(ha->uc.uniq_id)); +	eui[0] ^= 2; +	return 0; +} +  static int addrconf_ifid_arcnet(u8 *eui, struct net_device *dev)  {  	/* XXX: inherit EUI-64 from other interface -- yoshfuji */  	if (dev->addr_len != ARCNET_ALEN)  		return -1;  	memset(eui, 0, 7); -	eui[7] = *(u8*)dev->dev_addr; +	eui[7] = *(u8 *)dev->dev_addr;  	return 0;  } @@ -1575,12 +1832,26 @@ static int addrconf_ifid_sit(u8 *eui, struct net_device *dev)  	return -1;  } +static int addrconf_ifid_gre(u8 *eui, struct net_device *dev) +{ +	return __ipv6_isatap_ifid(eui, *(__be32 *)dev->dev_addr); +} + +static int addrconf_ifid_ip6tnl(u8 *eui, struct net_device *dev) +{ +	memcpy(eui, dev->perm_addr, 3); +	memcpy(eui + 5, dev->perm_addr + 3, 3); +	eui[3] = 0xFF; +	eui[4] = 0xFE; +	eui[0] ^= 2; +	return 0; +} +  static int ipv6_generate_eui64(u8 *eui, struct net_device *dev)  {  	switch (dev->type) {  	case ARPHRD_ETHER:  	case ARPHRD_FDDI: -	case ARPHRD_IEEE802_TR:  		return addrconf_ifid_eui48(eui, dev);  	case ARPHRD_ARCNET:  		return addrconf_ifid_arcnet(eui, dev); @@ -1588,6 +1859,15 @@ static int ipv6_generate_eui64(u8 *eui, struct net_device *dev)  		return addrconf_ifid_infiniband(eui, dev);  	case ARPHRD_SIT:  		return addrconf_ifid_sit(eui, dev); +	case ARPHRD_IPGRE: +		return addrconf_ifid_gre(eui, dev); +	case ARPHRD_6LOWPAN: +	case ARPHRD_IEEE802154: +		return addrconf_ifid_eui64(eui, dev); +	case ARPHRD_IEEE1394: +		return addrconf_ifid_ieee1394(eui, dev); +	case ARPHRD_TUNNEL6: +		return addrconf_ifid_ip6tnl(eui, dev);  	}  	return -1;  } @@ -1598,7 +1878,9 @@ static int ipv6_inherit_eui64(u8 *eui, struct inet6_dev *idev)  	struct inet6_ifaddr *ifp;  	read_lock_bh(&idev->lock); -	list_for_each_entry(ifp, &idev->addr_list, if_list) { +	list_for_each_entry_reverse(ifp, &idev->addr_list, if_list) { +		if (ifp->scope > IFA_LINK) +			break;  		if (ifp->scope == IFA_LINK && !(ifp->flags&IFA_F_TENTATIVE)) {  			memcpy(eui, ifp->addr.s6_addr+8, 8);  			err = 0; @@ -1609,9 +1891,8 @@ static int ipv6_inherit_eui64(u8 *eui, struct inet6_dev *idev)  	return err;  } -#ifdef CONFIG_IPV6_PRIVACY  /* (re)generation of randomized interface identifier (RFC 3041 3.2, 3.5) */ -static int __ipv6_regen_rndid(struct inet6_dev *idev) +static void __ipv6_regen_rndid(struct inet6_dev *idev)  {  regen:  	get_random_bytes(idev->rndid, sizeof(idev->rndid)); @@ -1638,8 +1919,6 @@ regen:  		if ((idev->rndid[2]|idev->rndid[3]|idev->rndid[4]|idev->rndid[5]|idev->rndid[6]|idev->rndid[7]) == 0x00)  			goto regen;  	} - -	return 0;  }  static void ipv6_regen_rndid(unsigned long data) @@ -1653,17 +1932,16 @@ static void ipv6_regen_rndid(unsigned long data)  	if (idev->dead)  		goto out; -	if (__ipv6_regen_rndid(idev) < 0) -		goto out; +	__ipv6_regen_rndid(idev);  	expires = jiffies +  		idev->cnf.temp_prefered_lft * HZ - -		idev->cnf.regen_max_retry * idev->cnf.dad_transmits * idev->nd_parms->retrans_time - +		idev->cnf.regen_max_retry * idev->cnf.dad_transmits * +		NEIGH_VAR(idev->nd_parms, RETRANS_TIME) -  		idev->cnf.max_desync_factor * HZ;  	if (time_before(expires, jiffies)) { -		printk(KERN_WARNING -			"ipv6_regen_rndid(): too short regeneration interval; timer disabled for %s.\n", -			idev->dev->name); +		pr_warn("%s: too short regeneration interval; timer disabled for %s\n", +			__func__, idev->dev->name);  		goto out;  	} @@ -1676,14 +1954,11 @@ out:  	in6_dev_put(idev);  } -static int __ipv6_try_regen_rndid(struct inet6_dev *idev, struct in6_addr *tmpaddr) { -	int ret = 0; - +static void  __ipv6_try_regen_rndid(struct inet6_dev *idev, struct in6_addr *tmpaddr) +{  	if (tmpaddr && memcmp(idev->rndid, &tmpaddr->s6_addr[8], 8) == 0) -		ret = __ipv6_regen_rndid(idev); -	return ret; +		__ipv6_regen_rndid(idev);  } -#endif  /*   *	Add prefix route. @@ -1704,13 +1979,13 @@ addrconf_prefix_route(struct in6_addr *pfx, int plen, struct net_device *dev,  		.fc_protocol = RTPROT_KERNEL,  	}; -	ipv6_addr_copy(&cfg.fc_dst, pfx); +	cfg.fc_dst = *pfx;  	/* Prevent useless cloning on PtP SIT.  	   This thing is done here expecting that the whole  	   class of non-broadcast devices need not cloning.  	 */ -#if defined(CONFIG_IPV6_SIT) || defined(CONFIG_IPV6_SIT_MODULE) +#if IS_ENABLED(CONFIG_IPV6_SIT)  	if (dev->type == ARPHRD_SIT && (dev->flags & IFF_POINTOPOINT))  		cfg.fc_flags |= RTF_NONEXTHOP;  #endif @@ -1718,6 +1993,40 @@ addrconf_prefix_route(struct in6_addr *pfx, int plen, struct net_device *dev,  	ip6_route_add(&cfg);  } + +static struct rt6_info *addrconf_get_prefix_route(const struct in6_addr *pfx, +						  int plen, +						  const struct net_device *dev, +						  u32 flags, u32 noflags) +{ +	struct fib6_node *fn; +	struct rt6_info *rt = NULL; +	struct fib6_table *table; + +	table = fib6_get_table(dev_net(dev), RT6_TABLE_PREFIX); +	if (table == NULL) +		return NULL; + +	read_lock_bh(&table->tb6_lock); +	fn = fib6_locate(&table->tb6_root, pfx, plen, NULL, 0); +	if (!fn) +		goto out; +	for (rt = fn->leaf; rt; rt = rt->dst.rt6_next) { +		if (rt->dst.dev->ifindex != dev->ifindex) +			continue; +		if ((rt->rt6i_flags & flags) != flags) +			continue; +		if ((rt->rt6i_flags & noflags) != 0) +			continue; +		dst_hold(&rt->dst); +		break; +	} +out: +	read_unlock_bh(&table->tb6_lock); +	return rt; +} + +  /* Create "default" multicast route to the interface */  static void addrconf_add_mroute(struct net_device *dev) @@ -1736,31 +2045,6 @@ static void addrconf_add_mroute(struct net_device *dev)  	ip6_route_add(&cfg);  } -#if defined(CONFIG_IPV6_SIT) || defined(CONFIG_IPV6_SIT_MODULE) -static void sit_route_add(struct net_device *dev) -{ -	struct fib6_config cfg = { -		.fc_table = RT6_TABLE_MAIN, -		.fc_metric = IP6_RT_PRIO_ADDRCONF, -		.fc_ifindex = dev->ifindex, -		.fc_dst_len = 96, -		.fc_flags = RTF_UP | RTF_NONEXTHOP, -		.fc_nlinfo.nl_net = dev_net(dev), -	}; - -	/* prefix length - 96 bits "::d.d.d.d" */ -	ip6_route_add(&cfg); -} -#endif - -static void addrconf_add_lroute(struct net_device *dev) -{ -	struct in6_addr addr; - -	ipv6_addr_set(&addr,  htonl(0xFE800000), 0, 0, 0); -	addrconf_prefix_route(&addr, 64, dev, 0, 0); -} -  static struct inet6_dev *addrconf_add_dev(struct net_device *dev)  {  	struct inet6_dev *idev; @@ -1775,14 +2059,80 @@ static struct inet6_dev *addrconf_add_dev(struct net_device *dev)  		return ERR_PTR(-EACCES);  	/* Add default multicast route */ -	addrconf_add_mroute(dev); +	if (!(dev->flags & IFF_LOOPBACK)) +		addrconf_add_mroute(dev); -	/* Add link local route */ -	addrconf_add_lroute(dev);  	return idev;  } -void addrconf_prefix_rcv(struct net_device *dev, u8 *opt, int len) +static void manage_tempaddrs(struct inet6_dev *idev, +			     struct inet6_ifaddr *ifp, +			     __u32 valid_lft, __u32 prefered_lft, +			     bool create, unsigned long now) +{ +	u32 flags; +	struct inet6_ifaddr *ift; + +	read_lock_bh(&idev->lock); +	/* update all temporary addresses in the list */ +	list_for_each_entry(ift, &idev->tempaddr_list, tmp_list) { +		int age, max_valid, max_prefered; + +		if (ifp != ift->ifpub) +			continue; + +		/* RFC 4941 section 3.3: +		 * If a received option will extend the lifetime of a public +		 * address, the lifetimes of temporary addresses should +		 * be extended, subject to the overall constraint that no +		 * temporary addresses should ever remain "valid" or "preferred" +		 * for a time longer than (TEMP_VALID_LIFETIME) or +		 * (TEMP_PREFERRED_LIFETIME - DESYNC_FACTOR), respectively. +		 */ +		age = (now - ift->cstamp) / HZ; +		max_valid = idev->cnf.temp_valid_lft - age; +		if (max_valid < 0) +			max_valid = 0; + +		max_prefered = idev->cnf.temp_prefered_lft - +			       idev->cnf.max_desync_factor - age; +		if (max_prefered < 0) +			max_prefered = 0; + +		if (valid_lft > max_valid) +			valid_lft = max_valid; + +		if (prefered_lft > max_prefered) +			prefered_lft = max_prefered; + +		spin_lock(&ift->lock); +		flags = ift->flags; +		ift->valid_lft = valid_lft; +		ift->prefered_lft = prefered_lft; +		ift->tstamp = now; +		if (prefered_lft > 0) +			ift->flags &= ~IFA_F_DEPRECATED; + +		spin_unlock(&ift->lock); +		if (!(flags&IFA_F_TENTATIVE)) +			ipv6_ifa_notify(0, ift); +	} + +	if ((create || list_empty(&idev->tempaddr_list)) && +	    idev->cnf.use_tempaddr > 0) { +		/* When a new public address is created as described +		 * in [ADDRCONF], also create a new temporary address. +		 * Also create a temporary address if it's enabled but +		 * no temporary address currently exists. +		 */ +		read_unlock_bh(&idev->lock); +		ipv6_create_tempaddr(ifp, NULL); +	} else { +		read_unlock_bh(&idev->lock); +	} +} + +void addrconf_prefix_rcv(struct net_device *dev, u8 *opt, int len, bool sllao)  {  	struct prefix_info *pinfo;  	__u32 valid_lft; @@ -1794,7 +2144,7 @@ void addrconf_prefix_rcv(struct net_device *dev, u8 *opt, int len)  	pinfo = (struct prefix_info *) opt;  	if (len < sizeof(struct prefix_info)) { -		ADBG(("addrconf: prefix option too short\n")); +		ADBG("addrconf: prefix option too short\n");  		return;  	} @@ -1811,16 +2161,15 @@ void addrconf_prefix_rcv(struct net_device *dev, u8 *opt, int len)  	prefered_lft = ntohl(pinfo->prefered);  	if (prefered_lft > valid_lft) { -		if (net_ratelimit()) -			printk(KERN_WARNING "addrconf: prefix option has invalid lifetime\n"); +		net_warn_ratelimited("addrconf: prefix option has invalid lifetime\n");  		return;  	}  	in6_dev = in6_dev_get(dev);  	if (in6_dev == NULL) { -		if (net_ratelimit()) -			printk(KERN_DEBUG "addrconf: device %s not configured\n", dev->name); +		net_dbg_ratelimited("addrconf: device %s not configured\n", +				    dev->name);  		return;  	} @@ -1847,21 +2196,22 @@ void addrconf_prefix_rcv(struct net_device *dev, u8 *opt, int len)  		if (addrconf_finite_timeout(rt_expires))  			rt_expires *= HZ; -		rt = rt6_lookup(net, &pinfo->prefix, NULL, -				dev->ifindex, 1); +		rt = addrconf_get_prefix_route(&pinfo->prefix, +					       pinfo->prefix_len, +					       dev, +					       RTF_ADDRCONF | RTF_PREFIX_RT, +					       RTF_GATEWAY | RTF_DEFAULT); -		if (rt && addrconf_is_prefix_route(rt)) { +		if (rt) {  			/* Autoconf prefix route */  			if (valid_lft == 0) {  				ip6_del_rt(rt);  				rt = NULL;  			} else if (addrconf_finite_timeout(rt_expires)) {  				/* not infinity */ -				rt->rt6i_expires = jiffies + rt_expires; -				rt->rt6i_flags |= RTF_EXPIRES; +				rt6_set_expires(rt, jiffies + rt_expires);  			} else { -				rt->rt6i_flags &= ~RTF_EXPIRES; -				rt->rt6i_expires = 0; +				rt6_clean_expires(rt);  			}  		} else if (valid_lft) {  			clock_t expires = 0; @@ -1874,29 +2224,35 @@ void addrconf_prefix_rcv(struct net_device *dev, u8 *opt, int len)  			addrconf_prefix_route(&pinfo->prefix, pinfo->prefix_len,  					      dev, expires, flags);  		} -		if (rt) -			dst_release(&rt->dst); +		ip6_rt_put(rt);  	}  	/* Try to figure out our local address for this prefix */  	if (pinfo->autoconf && in6_dev->cnf.autoconf) { -		struct inet6_ifaddr * ifp; +		struct inet6_ifaddr *ifp;  		struct in6_addr addr;  		int create = 0, update_lft = 0; +		bool tokenized = false;  		if (pinfo->prefix_len == 64) {  			memcpy(&addr, &pinfo->prefix, 8); -			if (ipv6_generate_eui64(addr.s6_addr + 8, dev) && -			    ipv6_inherit_eui64(addr.s6_addr + 8, in6_dev)) { + +			if (!ipv6_addr_any(&in6_dev->token)) { +				read_lock_bh(&in6_dev->lock); +				memcpy(addr.s6_addr + 8, +				       in6_dev->token.s6_addr + 8, 8); +				read_unlock_bh(&in6_dev->lock); +				tokenized = true; +			} else if (ipv6_generate_eui64(addr.s6_addr + 8, dev) && +				   ipv6_inherit_eui64(addr.s6_addr + 8, in6_dev)) {  				in6_dev_put(in6_dev);  				return;  			}  			goto ok;  		} -		if (net_ratelimit()) -			printk(KERN_DEBUG "IPv6 addrconf: prefix with wrong length %d\n", -			       pinfo->prefix_len); +		net_dbg_ratelimited("IPv6 addrconf: prefix with wrong length %d\n", +				    pinfo->prefix_len);  		in6_dev_put(in6_dev);  		return; @@ -1910,7 +2266,7 @@ ok:  #ifdef CONFIG_IPV6_OPTIMISTIC_DAD  			if (in6_dev->cnf.optimistic_dad && -			    !net->ipv6.devconf_all->forwarding) +			    !net->ipv6.devconf_all->forwarding && sllao)  				addr_flags = IFA_F_OPTIMISTIC;  #endif @@ -1919,26 +2275,30 @@ ok:  			 */  			if (!max_addresses ||  			    ipv6_count_addresses(in6_dev) < max_addresses) -				ifp = ipv6_add_addr(in6_dev, &addr, pinfo->prefix_len, +				ifp = ipv6_add_addr(in6_dev, &addr, NULL, +						    pinfo->prefix_len,  						    addr_type&IPV6_ADDR_SCOPE_MASK, -						    addr_flags); +						    addr_flags, valid_lft, +						    prefered_lft); -			if (!ifp || IS_ERR(ifp)) { +			if (IS_ERR_OR_NULL(ifp)) {  				in6_dev_put(in6_dev);  				return;  			} -			update_lft = create = 1; +			update_lft = 0; +			create = 1; +			spin_lock_bh(&ifp->lock); +			ifp->flags |= IFA_F_MANAGETEMPADDR;  			ifp->cstamp = jiffies; -			addrconf_dad_start(ifp, RTF_ADDRCONF|RTF_PREFIX_RT); +			ifp->tokenized = tokenized; +			spin_unlock_bh(&ifp->lock); +			addrconf_dad_start(ifp);  		}  		if (ifp) { -			int flags; +			u32 flags;  			unsigned long now; -#ifdef CONFIG_IPV6_PRIVACY -			struct inet6_ifaddr *ift; -#endif  			u32 stored_lft;  			/* update lifetime (RFC2462 5.5.3 e) */ @@ -1948,44 +2308,22 @@ ok:  				stored_lft = ifp->valid_lft - (now - ifp->tstamp) / HZ;  			else  				stored_lft = 0; -			if (!update_lft && stored_lft) { -				if (valid_lft > MIN_VALID_LIFETIME || -				    valid_lft > stored_lft) -					update_lft = 1; -				else if (stored_lft <= MIN_VALID_LIFETIME) { -					/* valid_lft <= stored_lft is always true */ -					/* -					 * RFC 4862 Section 5.5.3e: -					 * "Note that the preferred lifetime of -					 *  the corresponding address is always -					 *  reset to the Preferred Lifetime in -					 *  the received Prefix Information -					 *  option, regardless of whether the -					 *  valid lifetime is also reset or -					 *  ignored." -					 * -					 *  So if the preferred lifetime in -					 *  this advertisement is different -					 *  than what we have stored, but the -					 *  valid lifetime is invalid, just -					 *  reset prefered_lft. -					 * -					 *  We must set the valid lifetime -					 *  to the stored lifetime since we'll -					 *  be updating the timestamp below, -					 *  else we'll set it back to the -					 *  minumum. -					 */ -					if (prefered_lft != ifp->prefered_lft) { -						valid_lft = stored_lft; -						update_lft = 1; -					} -				} else { -					valid_lft = MIN_VALID_LIFETIME; -					if (valid_lft < prefered_lft) -						prefered_lft = valid_lft; -					update_lft = 1; -				} +			if (!update_lft && !create && stored_lft) { +				const u32 minimum_lft = min( +					stored_lft, (u32)MIN_VALID_LIFETIME); +				valid_lft = max(valid_lft, minimum_lft); + +				/* RFC4862 Section 5.5.3e: +				 * "Note that the preferred lifetime of the +				 *  corresponding address is always reset to +				 *  the Preferred Lifetime in the received +				 *  Prefix Information option, regardless of +				 *  whether the valid lifetime is also reset or +				 *  ignored." +				 * +				 * So we should always update prefered_lft here. +				 */ +				update_lft = 1;  			}  			if (update_lft) { @@ -2001,47 +2339,11 @@ ok:  			} else  				spin_unlock(&ifp->lock); -#ifdef CONFIG_IPV6_PRIVACY -			read_lock_bh(&in6_dev->lock); -			/* update all temporary addresses in the list */ -			list_for_each_entry(ift, &in6_dev->tempaddr_list, tmp_list) { -				/* -				 * When adjusting the lifetimes of an existing -				 * temporary address, only lower the lifetimes. -				 * Implementations must not increase the -				 * lifetimes of an existing temporary address -				 * when processing a Prefix Information Option. -				 */ -				if (ifp != ift->ifpub) -					continue; - -				spin_lock(&ift->lock); -				flags = ift->flags; -				if (ift->valid_lft > valid_lft && -				    ift->valid_lft - valid_lft > (jiffies - ift->tstamp) / HZ) -					ift->valid_lft = valid_lft + (jiffies - ift->tstamp) / HZ; -				if (ift->prefered_lft > prefered_lft && -				    ift->prefered_lft - prefered_lft > (jiffies - ift->tstamp) / HZ) -					ift->prefered_lft = prefered_lft + (jiffies - ift->tstamp) / HZ; -				spin_unlock(&ift->lock); -				if (!(flags&IFA_F_TENTATIVE)) -					ipv6_ifa_notify(0, ift); -			} +			manage_tempaddrs(in6_dev, ifp, valid_lft, prefered_lft, +					 create, now); -			if ((create || list_empty(&in6_dev->tempaddr_list)) && in6_dev->cnf.use_tempaddr > 0) { -				/* -				 * When a new public address is created as described in [ADDRCONF], -				 * also create a new temporary address. Also create a temporary -				 * address if it's enabled but no temporary address currently exists. -				 */ -				read_unlock_bh(&in6_dev->lock); -				ipv6_create_tempaddr(ifp, NULL); -			} else { -				read_unlock_bh(&in6_dev->lock); -			} -#endif  			in6_ifa_put(ifp); -			addrconf_verify(0); +			addrconf_verify();  		}  	}  	inet6_prefix_notify(RTM_NEWPREFIX, in6_dev, pinfo); @@ -2071,7 +2373,7 @@ int addrconf_set_dstaddr(struct net *net, void __user *arg)  	if (dev == NULL)  		goto err_exit; -#if defined(CONFIG_IPV6_SIT) || defined(CONFIG_IPV6_SIT_MODULE) +#if IS_ENABLED(CONFIG_IPV6_SIT)  	if (dev->type == ARPHRD_SIT) {  		const struct net_device_ops *ops = dev->netdev_ops;  		struct ifreq ifr; @@ -2117,9 +2419,11 @@ err_exit:  /*   *	Manual configuration of address on an interface   */ -static int inet6_addr_add(struct net *net, int ifindex, struct in6_addr *pfx, -			  unsigned int plen, __u8 ifa_flags, __u32 prefered_lft, -			  __u32 valid_lft) +static int inet6_addr_add(struct net *net, int ifindex, +			  const struct in6_addr *pfx, +			  const struct in6_addr *peer_pfx, +			  unsigned int plen, __u32 ifa_flags, +			  __u32 prefered_lft, __u32 valid_lft)  {  	struct inet6_ifaddr *ifp;  	struct inet6_dev *idev; @@ -2138,6 +2442,9 @@ static int inet6_addr_add(struct net *net, int ifindex, struct in6_addr *pfx,  	if (!valid_lft || prefered_lft > valid_lft)  		return -EINVAL; +	if (ifa_flags & IFA_F_MANAGETEMPADDR && plen != 64) +		return -EINVAL; +  	dev = __dev_get_by_index(net, ifindex);  	if (!dev)  		return -ENODEV; @@ -2166,33 +2473,34 @@ static int inet6_addr_add(struct net *net, int ifindex, struct in6_addr *pfx,  		prefered_lft = timeout;  	} -	ifp = ipv6_add_addr(idev, pfx, plen, scope, ifa_flags); +	ifp = ipv6_add_addr(idev, pfx, peer_pfx, plen, scope, ifa_flags, +			    valid_lft, prefered_lft);  	if (!IS_ERR(ifp)) { -		spin_lock_bh(&ifp->lock); -		ifp->valid_lft = valid_lft; -		ifp->prefered_lft = prefered_lft; -		ifp->tstamp = jiffies; -		spin_unlock_bh(&ifp->lock); +		if (!(ifa_flags & IFA_F_NOPREFIXROUTE)) { +			addrconf_prefix_route(&ifp->addr, ifp->prefix_len, dev, +					      expires, flags); +		} -		addrconf_prefix_route(&ifp->addr, ifp->prefix_len, dev, -				      expires, flags);  		/*  		 * Note that section 3.1 of RFC 4429 indicates  		 * that the Optimistic flag should not be set for  		 * manually configured addresses  		 */ -		addrconf_dad_start(ifp, 0); +		addrconf_dad_start(ifp); +		if (ifa_flags & IFA_F_MANAGETEMPADDR) +			manage_tempaddrs(idev, ifp, valid_lft, prefered_lft, +					 true, jiffies);  		in6_ifa_put(ifp); -		addrconf_verify(0); +		addrconf_verify_rtnl();  		return 0;  	}  	return PTR_ERR(ifp);  } -static int inet6_addr_del(struct net *net, int ifindex, struct in6_addr *pfx, -			  unsigned int plen) +static int inet6_addr_del(struct net *net, int ifindex, u32 ifa_flags, +			  const struct in6_addr *pfx, unsigned int plen)  {  	struct inet6_ifaddr *ifp;  	struct inet6_dev *idev; @@ -2215,13 +2523,12 @@ static int inet6_addr_del(struct net *net, int ifindex, struct in6_addr *pfx,  			in6_ifa_hold(ifp);  			read_unlock_bh(&idev->lock); +			if (!(ifp->flags & IFA_F_TEMPORARY) && +			    (ifa_flags & IFA_F_MANAGETEMPADDR)) +				manage_tempaddrs(idev, ifp, 0, 0, false, +						 jiffies);  			ipv6_del_addr(ifp); - -			/* If the last address is deleted administratively, -			   disable IPv6 on this interface. -			 */ -			if (list_empty(&idev->addr_list)) -				addrconf_ifdown(idev->dev, 1); +			addrconf_verify_rtnl();  			return 0;  		}  	} @@ -2235,14 +2542,14 @@ int addrconf_add_ifaddr(struct net *net, void __user *arg)  	struct in6_ifreq ireq;  	int err; -	if (!capable(CAP_NET_ADMIN)) +	if (!ns_capable(net->user_ns, CAP_NET_ADMIN))  		return -EPERM;  	if (copy_from_user(&ireq, arg, sizeof(struct in6_ifreq)))  		return -EFAULT;  	rtnl_lock(); -	err = inet6_addr_add(net, ireq.ifr6_ifindex, &ireq.ifr6_addr, +	err = inet6_addr_add(net, ireq.ifr6_ifindex, &ireq.ifr6_addr, NULL,  			     ireq.ifr6_prefixlen, IFA_F_PERMANENT,  			     INFINITY_LIFE_TIME, INFINITY_LIFE_TIME);  	rtnl_unlock(); @@ -2254,14 +2561,14 @@ int addrconf_del_ifaddr(struct net *net, void __user *arg)  	struct in6_ifreq ireq;  	int err; -	if (!capable(CAP_NET_ADMIN)) +	if (!ns_capable(net->user_ns, CAP_NET_ADMIN))  		return -EPERM;  	if (copy_from_user(&ireq, arg, sizeof(struct in6_ifreq)))  		return -EFAULT;  	rtnl_lock(); -	err = inet6_addr_del(net, ireq.ifr6_ifindex, &ireq.ifr6_addr, +	err = inet6_addr_del(net, ireq.ifr6_ifindex, 0, &ireq.ifr6_addr,  			     ireq.ifr6_prefixlen);  	rtnl_unlock();  	return err; @@ -2272,7 +2579,9 @@ static void add_addr(struct inet6_dev *idev, const struct in6_addr *addr,  {  	struct inet6_ifaddr *ifp; -	ifp = ipv6_add_addr(idev, addr, plen, scope, IFA_F_PERMANENT); +	ifp = ipv6_add_addr(idev, addr, NULL, plen, +			    scope, IFA_F_PERMANENT, +			    INFINITY_LIFE_TIME, INFINITY_LIFE_TIME);  	if (!IS_ERR(ifp)) {  		spin_lock_bh(&ifp->lock);  		ifp->flags &= ~IFA_F_TENTATIVE; @@ -2282,13 +2591,14 @@ static void add_addr(struct inet6_dev *idev, const struct in6_addr *addr,  	}  } -#if defined(CONFIG_IPV6_SIT) || defined(CONFIG_IPV6_SIT_MODULE) +#if IS_ENABLED(CONFIG_IPV6_SIT)  static void sit_add_v4_addrs(struct inet6_dev *idev)  {  	struct in6_addr addr;  	struct net_device *dev;  	struct net *net = dev_net(idev->dev); -	int scope; +	int scope, plen; +	u32 pflags = 0;  	ASSERT_RTNL(); @@ -2298,24 +2608,27 @@ static void sit_add_v4_addrs(struct inet6_dev *idev)  	if (idev->dev->flags&IFF_POINTOPOINT) {  		addr.s6_addr32[0] = htonl(0xfe800000);  		scope = IFA_LINK; +		plen = 64;  	} else {  		scope = IPV6_ADDR_COMPATv4; +		plen = 96; +		pflags |= RTF_NONEXTHOP;  	}  	if (addr.s6_addr32[3]) { -		add_addr(idev, &addr, 128, scope); +		add_addr(idev, &addr, plen, scope); +		addrconf_prefix_route(&addr, plen, idev->dev, 0, pflags);  		return;  	}  	for_each_netdev(net, dev) { -		struct in_device * in_dev = __in_dev_get_rtnl(dev); +		struct in_device *in_dev = __in_dev_get_rtnl(dev);  		if (in_dev && (dev->flags & IFF_UP)) { -			struct in_ifaddr * ifa; +			struct in_ifaddr *ifa;  			int flag = scope;  			for (ifa = in_dev->ifa_list; ifa; ifa = ifa->ifa_next) { -				int plen;  				addr.s6_addr32[3] = ifa->ifa_local; @@ -2326,12 +2639,10 @@ static void sit_add_v4_addrs(struct inet6_dev *idev)  						continue;  					flag |= IFA_HOST;  				} -				if (idev->dev->flags&IFF_POINTOPOINT) -					plen = 64; -				else -					plen = 96;  				add_addr(idev, &addr, plen, flag); +				addrconf_prefix_route(&addr, plen, idev->dev, 0, +						      pflags);  			}  		}  	} @@ -2341,22 +2652,64 @@ static void sit_add_v4_addrs(struct inet6_dev *idev)  static void init_loopback(struct net_device *dev)  {  	struct inet6_dev  *idev; +	struct net_device *sp_dev; +	struct inet6_ifaddr *sp_ifa; +	struct rt6_info *sp_rt;  	/* ::1 */  	ASSERT_RTNL();  	if ((idev = ipv6_find_idev(dev)) == NULL) { -		printk(KERN_DEBUG "init loopback: add_dev failed\n"); +		pr_debug("%s: add_dev failed\n", __func__);  		return;  	}  	add_addr(idev, &in6addr_loopback, 128, IFA_HOST); + +	/* Add routes to other interface's IPv6 addresses */ +	for_each_netdev(dev_net(dev), sp_dev) { +		if (!strcmp(sp_dev->name, dev->name)) +			continue; + +		idev = __in6_dev_get(sp_dev); +		if (!idev) +			continue; + +		read_lock_bh(&idev->lock); +		list_for_each_entry(sp_ifa, &idev->addr_list, if_list) { + +			if (sp_ifa->flags & (IFA_F_DADFAILED | IFA_F_TENTATIVE)) +				continue; + +			if (sp_ifa->rt) { +				/* This dst has been added to garbage list when +				 * lo device down, release this obsolete dst and +				 * reallocate a new router for ifa. +				 */ +				if (sp_ifa->rt->dst.obsolete > 0) { +					ip6_rt_put(sp_ifa->rt); +					sp_ifa->rt = NULL; +				} else { +					continue; +				} +			} + +			sp_rt = addrconf_dst_alloc(idev, &sp_ifa->addr, false); + +			/* Failure cases are ignored */ +			if (!IS_ERR(sp_rt)) { +				sp_ifa->rt = sp_rt; +				ip6_ins_rt(sp_rt); +			} +		} +		read_unlock_bh(&idev->lock); +	}  } -static void addrconf_add_linklocal(struct inet6_dev *idev, struct in6_addr *addr) +static void addrconf_add_linklocal(struct inet6_dev *idev, const struct in6_addr *addr)  { -	struct inet6_ifaddr * ifp; +	struct inet6_ifaddr *ifp;  	u32 addr_flags = IFA_F_PERMANENT;  #ifdef CONFIG_IPV6_OPTIMISTIC_DAD @@ -2366,10 +2719,11 @@ static void addrconf_add_linklocal(struct inet6_dev *idev, struct in6_addr *addr  #endif -	ifp = ipv6_add_addr(idev, addr, 64, IFA_LINK, addr_flags); +	ifp = ipv6_add_addr(idev, addr, NULL, 64, IFA_LINK, addr_flags, +			    INFINITY_LIFE_TIME, INFINITY_LIFE_TIME);  	if (!IS_ERR(ifp)) {  		addrconf_prefix_route(&ifp->addr, ifp->prefix_len, idev->dev, 0, 0); -		addrconf_dad_start(ifp, 0); +		addrconf_dad_start(ifp);  		in6_ifa_put(ifp);  	}  } @@ -2377,15 +2731,18 @@ static void addrconf_add_linklocal(struct inet6_dev *idev, struct in6_addr *addr  static void addrconf_dev_config(struct net_device *dev)  {  	struct in6_addr addr; -	struct inet6_dev    * idev; +	struct inet6_dev *idev;  	ASSERT_RTNL();  	if ((dev->type != ARPHRD_ETHER) &&  	    (dev->type != ARPHRD_FDDI) && -	    (dev->type != ARPHRD_IEEE802_TR) &&  	    (dev->type != ARPHRD_ARCNET) && -	    (dev->type != ARPHRD_INFINIBAND)) { +	    (dev->type != ARPHRD_INFINIBAND) && +	    (dev->type != ARPHRD_IEEE802154) && +	    (dev->type != ARPHRD_IEEE1394) && +	    (dev->type != ARPHRD_TUNNEL6) && +	    (dev->type != ARPHRD_6LOWPAN)) {  		/* Alas, we support only Ethernet autoconfiguration. */  		return;  	} @@ -2401,7 +2758,7 @@ static void addrconf_dev_config(struct net_device *dev)  		addrconf_add_linklocal(idev, &addr);  } -#if defined(CONFIG_IPV6_SIT) || defined(CONFIG_IPV6_SIT_MODULE) +#if IS_ENABLED(CONFIG_IPV6_SIT)  static void addrconf_sit_config(struct net_device *dev)  {  	struct inet6_dev *idev; @@ -2415,7 +2772,7 @@ static void addrconf_sit_config(struct net_device *dev)  	 */  	if ((idev = ipv6_find_idev(dev)) == NULL) { -		printk(KERN_DEBUG "init sit: add_dev failed\n"); +		pr_debug("%s: add_dev failed\n", __func__);  		return;  	} @@ -2423,7 +2780,6 @@ static void addrconf_sit_config(struct net_device *dev)  		struct in6_addr addr;  		ipv6_addr_set(&addr,  htonl(0xFE800000), 0, 0, 0); -		addrconf_prefix_route(&addr, 64, dev, 0, 0);  		if (!ipv6_generate_eui64(addr.s6_addr + 8, dev))  			addrconf_add_linklocal(idev, &addr);  		return; @@ -2431,68 +2787,36 @@ static void addrconf_sit_config(struct net_device *dev)  	sit_add_v4_addrs(idev); -	if (dev->flags&IFF_POINTOPOINT) { +	if (dev->flags&IFF_POINTOPOINT)  		addrconf_add_mroute(dev); -		addrconf_add_lroute(dev); -	} else -		sit_route_add(dev);  }  #endif -static inline int -ipv6_inherit_linklocal(struct inet6_dev *idev, struct net_device *link_dev) -{ -	struct in6_addr lladdr; - -	if (!ipv6_get_lladdr(link_dev, &lladdr, IFA_F_TENTATIVE)) { -		addrconf_add_linklocal(idev, &lladdr); -		return 0; -	} -	return -1; -} - -static void ip6_tnl_add_linklocal(struct inet6_dev *idev) -{ -	struct net_device *link_dev; -	struct net *net = dev_net(idev->dev); - -	/* first try to inherit the link-local address from the link device */ -	if (idev->dev->iflink && -	    (link_dev = __dev_get_by_index(net, idev->dev->iflink))) { -		if (!ipv6_inherit_linklocal(idev, link_dev)) -			return; -	} -	/* then try to inherit it from any device */ -	for_each_netdev(net, link_dev) { -		if (!ipv6_inherit_linklocal(idev, link_dev)) -			return; -	} -	printk(KERN_DEBUG "init ip6-ip6: add_linklocal failed\n"); -} - -/* - * Autoconfigure tunnel with a link-local address so routing protocols, - * DHCPv6, MLD etc. can be run over the virtual link - */ - -static void addrconf_ip6_tnl_config(struct net_device *dev) +#if IS_ENABLED(CONFIG_NET_IPGRE) +static void addrconf_gre_config(struct net_device *dev)  {  	struct inet6_dev *idev; +	struct in6_addr addr;  	ASSERT_RTNL(); -	idev = addrconf_add_dev(dev); -	if (IS_ERR(idev)) { -		printk(KERN_DEBUG "init ip6-ip6: add_dev failed\n"); +	if ((idev = ipv6_find_idev(dev)) == NULL) { +		pr_debug("%s: add_dev failed\n", __func__);  		return;  	} -	ip6_tnl_add_linklocal(idev); + +	ipv6_addr_set(&addr,  htonl(0xFE800000), 0, 0, 0); +	if (!ipv6_generate_eui64(addr.s6_addr + 8, dev)) +		addrconf_add_linklocal(idev, &addr); +	else +		addrconf_prefix_route(&addr, 64, dev, 0, 0);  } +#endif  static int addrconf_notify(struct notifier_block *this, unsigned long event, -			   void * data) +			   void *ptr)  { -	struct net_device *dev = (struct net_device *) data; +	struct net_device *dev = netdev_notifier_info_to_dev(ptr);  	struct inet6_dev *idev = __in6_dev_get(dev);  	int run_pending = 0;  	int err; @@ -2514,9 +2838,7 @@ static int addrconf_notify(struct notifier_block *this, unsigned long event,  		if (event == NETDEV_UP) {  			if (!addrconf_qdisc_ok(dev)) {  				/* device is not ready yet. */ -				printk(KERN_INFO -					"ADDRCONF(NETDEV_UP): %s: " -					"link is not ready\n", +				pr_info("ADDRCONF(NETDEV_UP): %s: link is not ready\n",  					dev->name);  				break;  			} @@ -2541,23 +2863,23 @@ static int addrconf_notify(struct notifier_block *this, unsigned long event,  				idev->if_flags |= IF_READY;  			} -			printk(KERN_INFO -					"ADDRCONF(NETDEV_CHANGE): %s: " -					"link becomes ready\n", -					dev->name); +			pr_info("ADDRCONF(NETDEV_CHANGE): %s: link becomes ready\n", +				dev->name);  			run_pending = 1;  		}  		switch (dev->type) { -#if defined(CONFIG_IPV6_SIT) || defined(CONFIG_IPV6_SIT_MODULE) +#if IS_ENABLED(CONFIG_IPV6_SIT)  		case ARPHRD_SIT:  			addrconf_sit_config(dev);  			break;  #endif -		case ARPHRD_TUNNEL6: -			addrconf_ip6_tnl_config(dev); +#if IS_ENABLED(CONFIG_NET_IPGRE) +		case ARPHRD_IPGRE: +			addrconf_gre_config(dev);  			break; +#endif  		case ARPHRD_LOOPBACK:  			init_loopback(dev);  			break; @@ -2607,7 +2929,7 @@ static int addrconf_notify(struct notifier_block *this, unsigned long event,  		}  		/* -		 * MTU falled under IPV6_MIN_MTU. +		 * if MTU under IPV6_MIN_MTU.  		 * Stop IPv6 on this interface.  		 */ @@ -2664,8 +2986,7 @@ static int addrconf_ifdown(struct net_device *dev, int how)  	struct net *net = dev_net(dev);  	struct inet6_dev *idev;  	struct inet6_ifaddr *ifa; -	LIST_HEAD(keep_list); -	int state; +	int state, i;  	ASSERT_RTNL(); @@ -2684,20 +3005,37 @@ static int addrconf_ifdown(struct net_device *dev, int how)  		idev->dead = 1;  		/* protected by rtnl_lock */ -		rcu_assign_pointer(dev->ip6_ptr, NULL); +		RCU_INIT_POINTER(dev->ip6_ptr, NULL);  		/* Step 1.5: remove snmp6 entry */  		snmp6_unregister_dev(idev);  	} +	/* Step 2: clear hash table */ +	for (i = 0; i < IN6_ADDR_HSIZE; i++) { +		struct hlist_head *h = &inet6_addr_lst[i]; + +		spin_lock_bh(&addrconf_hash_lock); +	restart: +		hlist_for_each_entry_rcu(ifa, h, addr_lst) { +			if (ifa->idev == idev) { +				hlist_del_init_rcu(&ifa->addr_lst); +				addrconf_del_dad_work(ifa); +				goto restart; +			} +		} +		spin_unlock_bh(&addrconf_hash_lock); +	} +  	write_lock_bh(&idev->lock); +	addrconf_del_rs_timer(idev); +  	/* Step 2: clear flags for stateless addrconf */  	if (!how)  		idev->if_flags &= ~(IF_RS_SENT|IF_RA_RCVD|IF_READY); -#ifdef CONFIG_IPV6_PRIVACY  	if (how && del_timer(&idev->regen_timer))  		in6_dev_put(idev); @@ -2717,59 +3055,29 @@ static int addrconf_ifdown(struct net_device *dev, int how)  		in6_ifa_put(ifa);  		write_lock_bh(&idev->lock);  	} -#endif  	while (!list_empty(&idev->addr_list)) {  		ifa = list_first_entry(&idev->addr_list,  				       struct inet6_ifaddr, if_list); -		addrconf_del_timer(ifa); - -		/* If just doing link down, and address is permanent -		   and not link-local, then retain it. */ -		if (!how && -		    (ifa->flags&IFA_F_PERMANENT) && -		    !(ipv6_addr_type(&ifa->addr) & IPV6_ADDR_LINKLOCAL)) { -			list_move_tail(&ifa->if_list, &keep_list); - -			/* If not doing DAD on this address, just keep it. */ -			if ((dev->flags&(IFF_NOARP|IFF_LOOPBACK)) || -			    idev->cnf.accept_dad <= 0 || -			    (ifa->flags & IFA_F_NODAD)) -				continue; +		addrconf_del_dad_work(ifa); -			/* If it was tentative already, no need to notify */ -			if (ifa->flags & IFA_F_TENTATIVE) -				continue; +		list_del(&ifa->if_list); -			/* Flag it for later restoration when link comes up */ -			ifa->flags |= IFA_F_TENTATIVE; -			ifa->state = INET6_IFADDR_STATE_DAD; -		} else { -			list_del(&ifa->if_list); - -			/* clear hash table */ -			spin_lock_bh(&addrconf_hash_lock); -			hlist_del_init_rcu(&ifa->addr_lst); -			spin_unlock_bh(&addrconf_hash_lock); - -			write_unlock_bh(&idev->lock); -			spin_lock_bh(&ifa->state_lock); -			state = ifa->state; -			ifa->state = INET6_IFADDR_STATE_DEAD; -			spin_unlock_bh(&ifa->state_lock); - -			if (state != INET6_IFADDR_STATE_DEAD) { -				__ipv6_ifa_notify(RTM_DELADDR, ifa); -				atomic_notifier_call_chain(&inet6addr_chain, -							   NETDEV_DOWN, ifa); -			} +		write_unlock_bh(&idev->lock); + +		spin_lock_bh(&ifa->state_lock); +		state = ifa->state; +		ifa->state = INET6_IFADDR_STATE_DEAD; +		spin_unlock_bh(&ifa->state_lock); -			in6_ifa_put(ifa); -			write_lock_bh(&idev->lock); +		if (state != INET6_IFADDR_STATE_DEAD) { +			__ipv6_ifa_notify(RTM_DELADDR, ifa); +			inet6addr_notifier_call_chain(NETDEV_DOWN, ifa);  		} -	} +		in6_ifa_put(ifa); -	list_splice(&keep_list, &idev->addr_list); +		write_lock_bh(&idev->lock); +	}  	write_unlock_bh(&idev->lock); @@ -2793,43 +3101,47 @@ static int addrconf_ifdown(struct net_device *dev, int how)  static void addrconf_rs_timer(unsigned long data)  { -	struct inet6_ifaddr *ifp = (struct inet6_ifaddr *) data; -	struct inet6_dev *idev = ifp->idev; +	struct inet6_dev *idev = (struct inet6_dev *)data; +	struct net_device *dev = idev->dev; +	struct in6_addr lladdr; -	read_lock(&idev->lock); +	write_lock(&idev->lock);  	if (idev->dead || !(idev->if_flags & IF_READY))  		goto out; -	if (idev->cnf.forwarding) +	if (!ipv6_accept_ra(idev))  		goto out;  	/* Announcement received after solicitation was sent */  	if (idev->if_flags & IF_RA_RCVD)  		goto out; -	spin_lock(&ifp->lock); -	if (ifp->probes++ < idev->cnf.rtr_solicits) { -		/* The wait after the last probe can be shorter */ -		addrconf_mod_timer(ifp, AC_RS, -				   (ifp->probes == idev->cnf.rtr_solicits) ? -				   idev->cnf.rtr_solicit_delay : -				   idev->cnf.rtr_solicit_interval); -		spin_unlock(&ifp->lock); +	if (idev->rs_probes++ < idev->cnf.rtr_solicits) { +		write_unlock(&idev->lock); +		if (!ipv6_get_lladdr(dev, &lladdr, IFA_F_TENTATIVE)) +			ndisc_send_rs(dev, &lladdr, +				      &in6addr_linklocal_allrouters); +		else +			goto put; -		ndisc_send_rs(idev->dev, &ifp->addr, &in6addr_linklocal_allrouters); +		write_lock(&idev->lock); +		/* The wait after the last probe can be shorter */ +		addrconf_mod_rs_timer(idev, (idev->rs_probes == +					     idev->cnf.rtr_solicits) ? +				      idev->cnf.rtr_solicit_delay : +				      idev->cnf.rtr_solicit_interval);  	} else { -		spin_unlock(&ifp->lock);  		/*  		 * Note: we do not support deprecated "all on-link"  		 * assumption any longer.  		 */ -		printk(KERN_DEBUG "%s: no IPv6 routers present\n", -		       idev->dev->name); +		pr_debug("%s: no IPv6 routers present\n", idev->dev->name);  	}  out: -	read_unlock(&idev->lock); -	in6_ifa_put(ifp); +	write_unlock(&idev->lock); +put: +	in6_dev_put(idev);  }  /* @@ -2843,20 +3155,20 @@ static void addrconf_dad_kick(struct inet6_ifaddr *ifp)  	if (ifp->flags & IFA_F_OPTIMISTIC)  		rand_num = 0;  	else -		rand_num = net_random() % (idev->cnf.rtr_solicit_delay ? : 1); +		rand_num = prandom_u32() % (idev->cnf.rtr_solicit_delay ? : 1); -	ifp->probes = idev->cnf.dad_transmits; -	addrconf_mod_timer(ifp, AC_DAD, rand_num); +	ifp->dad_probes = idev->cnf.dad_transmits; +	addrconf_mod_dad_work(ifp, rand_num);  } -static void addrconf_dad_start(struct inet6_ifaddr *ifp, u32 flags) +static void addrconf_dad_begin(struct inet6_ifaddr *ifp)  {  	struct inet6_dev *idev = ifp->idev;  	struct net_device *dev = idev->dev;  	addrconf_join_solict(dev, &ifp->addr); -	net_srandom(ifp->addr.s6_addr32[3]); +	prandom_seed((__force u32) ifp->addr.s6_addr32[3]);  	read_lock_bh(&idev->lock);  	spin_lock(&ifp->lock); @@ -2901,57 +3213,124 @@ out:  	read_unlock_bh(&idev->lock);  } -static void addrconf_dad_timer(unsigned long data) +static void addrconf_dad_start(struct inet6_ifaddr *ifp) +{ +	bool begin_dad = false; + +	spin_lock_bh(&ifp->state_lock); +	if (ifp->state != INET6_IFADDR_STATE_DEAD) { +		ifp->state = INET6_IFADDR_STATE_PREDAD; +		begin_dad = true; +	} +	spin_unlock_bh(&ifp->state_lock); + +	if (begin_dad) +		addrconf_mod_dad_work(ifp, 0); +} + +static void addrconf_dad_work(struct work_struct *w)  { -	struct inet6_ifaddr *ifp = (struct inet6_ifaddr *) data; +	struct inet6_ifaddr *ifp = container_of(to_delayed_work(w), +						struct inet6_ifaddr, +						dad_work);  	struct inet6_dev *idev = ifp->idev;  	struct in6_addr mcaddr; -	if (!ifp->probes && addrconf_dad_end(ifp)) +	enum { +		DAD_PROCESS, +		DAD_BEGIN, +		DAD_ABORT, +	} action = DAD_PROCESS; + +	rtnl_lock(); + +	spin_lock_bh(&ifp->state_lock); +	if (ifp->state == INET6_IFADDR_STATE_PREDAD) { +		action = DAD_BEGIN; +		ifp->state = INET6_IFADDR_STATE_DAD; +	} else if (ifp->state == INET6_IFADDR_STATE_ERRDAD) { +		action = DAD_ABORT; +		ifp->state = INET6_IFADDR_STATE_POSTDAD; +	} +	spin_unlock_bh(&ifp->state_lock); + +	if (action == DAD_BEGIN) { +		addrconf_dad_begin(ifp); +		goto out; +	} else if (action == DAD_ABORT) { +		addrconf_dad_stop(ifp, 1); +		goto out; +	} + +	if (!ifp->dad_probes && addrconf_dad_end(ifp))  		goto out; -	read_lock(&idev->lock); +	write_lock_bh(&idev->lock);  	if (idev->dead || !(idev->if_flags & IF_READY)) { -		read_unlock(&idev->lock); +		write_unlock_bh(&idev->lock);  		goto out;  	}  	spin_lock(&ifp->lock);  	if (ifp->state == INET6_IFADDR_STATE_DEAD) {  		spin_unlock(&ifp->lock); -		read_unlock(&idev->lock); +		write_unlock_bh(&idev->lock);  		goto out;  	} -	if (ifp->probes == 0) { +	if (ifp->dad_probes == 0) {  		/*  		 * DAD was successful  		 */  		ifp->flags &= ~(IFA_F_TENTATIVE|IFA_F_OPTIMISTIC|IFA_F_DADFAILED);  		spin_unlock(&ifp->lock); -		read_unlock(&idev->lock); +		write_unlock_bh(&idev->lock);  		addrconf_dad_completed(ifp);  		goto out;  	} -	ifp->probes--; -	addrconf_mod_timer(ifp, AC_DAD, ifp->idev->nd_parms->retrans_time); +	ifp->dad_probes--; +	addrconf_mod_dad_work(ifp, +			      NEIGH_VAR(ifp->idev->nd_parms, RETRANS_TIME));  	spin_unlock(&ifp->lock); -	read_unlock(&idev->lock); +	write_unlock_bh(&idev->lock);  	/* send a neighbour solicitation for our addr */  	addrconf_addr_solict_mult(&ifp->addr, &mcaddr);  	ndisc_send_ns(ifp->idev->dev, NULL, &ifp->addr, &mcaddr, &in6addr_any);  out:  	in6_ifa_put(ifp); +	rtnl_unlock(); +} + +/* ifp->idev must be at least read locked */ +static bool ipv6_lonely_lladdr(struct inet6_ifaddr *ifp) +{ +	struct inet6_ifaddr *ifpiter; +	struct inet6_dev *idev = ifp->idev; + +	list_for_each_entry_reverse(ifpiter, &idev->addr_list, if_list) { +		if (ifpiter->scope > IFA_LINK) +			break; +		if (ifp != ifpiter && ifpiter->scope == IFA_LINK && +		    (ifpiter->flags & (IFA_F_PERMANENT|IFA_F_TENTATIVE| +				       IFA_F_OPTIMISTIC|IFA_F_DADFAILED)) == +		    IFA_F_PERMANENT) +			return false; +	} +	return true;  }  static void addrconf_dad_completed(struct inet6_ifaddr *ifp)  {  	struct net_device *dev = ifp->idev->dev; +	struct in6_addr lladdr; +	bool send_rs, send_mld; + +	addrconf_del_dad_work(ifp);  	/*  	 *	Configure the address for reception. Now it is valid. @@ -2959,27 +3338,42 @@ static void addrconf_dad_completed(struct inet6_ifaddr *ifp)  	ipv6_ifa_notify(RTM_NEWADDR, ifp); -	/* If added prefix is link local and forwarding is off, -	   start sending router solicitations. +	/* If added prefix is link local and we are prepared to process +	   router advertisements, start sending router solicitations. +	 */ + +	read_lock_bh(&ifp->idev->lock); +	send_mld = ifp->scope == IFA_LINK && ipv6_lonely_lladdr(ifp); +	send_rs = send_mld && +		  ipv6_accept_ra(ifp->idev) && +		  ifp->idev->cnf.rtr_solicits > 0 && +		  (dev->flags&IFF_LOOPBACK) == 0; +	read_unlock_bh(&ifp->idev->lock); + +	/* While dad is in progress mld report's source address is in6_addrany. +	 * Resend with proper ll now.  	 */ +	if (send_mld) +		ipv6_mc_dad_complete(ifp->idev); -	if ((ifp->idev->cnf.forwarding == 0 || -	     ifp->idev->cnf.forwarding == 2) && -	    ifp->idev->cnf.rtr_solicits > 0 && -	    (dev->flags&IFF_LOOPBACK) == 0 && -	    (ipv6_addr_type(&ifp->addr) & IPV6_ADDR_LINKLOCAL)) { +	if (send_rs) {  		/*  		 *	If a host as already performed a random delay  		 *	[...] as part of DAD [...] there is no need  		 *	to delay again before sending the first RS  		 */ -		ndisc_send_rs(ifp->idev->dev, &ifp->addr, &in6addr_linklocal_allrouters); +		if (ipv6_get_lladdr(dev, &lladdr, IFA_F_TENTATIVE)) +			return; +		ndisc_send_rs(dev, &lladdr, &in6addr_linklocal_allrouters); -		spin_lock_bh(&ifp->lock); -		ifp->probes = 1; +		write_lock_bh(&ifp->idev->lock); +		spin_lock(&ifp->lock); +		ifp->idev->rs_probes = 1;  		ifp->idev->if_flags |= IF_RS_SENT; -		addrconf_mod_timer(ifp, AC_RS, ifp->idev->cnf.rtr_solicit_interval); -		spin_unlock_bh(&ifp->lock); +		addrconf_mod_rs_timer(ifp->idev, +				      ifp->idev->cnf.rtr_solicit_interval); +		spin_unlock(&ifp->lock); +		write_unlock_bh(&ifp->idev->lock);  	}  } @@ -3002,20 +3396,39 @@ static void addrconf_dad_run(struct inet6_dev *idev)  struct if6_iter_state {  	struct seq_net_private p;  	int bucket; +	int offset;  }; -static struct inet6_ifaddr *if6_get_first(struct seq_file *seq) +static struct inet6_ifaddr *if6_get_first(struct seq_file *seq, loff_t pos)  {  	struct inet6_ifaddr *ifa = NULL;  	struct if6_iter_state *state = seq->private;  	struct net *net = seq_file_net(seq); +	int p = 0; + +	/* initial bucket if pos is 0 */ +	if (pos == 0) { +		state->bucket = 0; +		state->offset = 0; +	} + +	for (; state->bucket < IN6_ADDR_HSIZE; ++state->bucket) { +		hlist_for_each_entry_rcu_bh(ifa, &inet6_addr_lst[state->bucket], +					 addr_lst) { +			if (!net_eq(dev_net(ifa->idev->dev), net)) +				continue; +			/* sync with offset */ +			if (p < state->offset) { +				p++; +				continue; +			} +			state->offset++; +			return ifa; +		} -	for (state->bucket = 0; state->bucket < IN6_ADDR_HSIZE; ++state->bucket) { -		struct hlist_node *n; -		hlist_for_each_entry_rcu_bh(ifa, n, &inet6_addr_lst[state->bucket], -					 addr_lst) -			if (net_eq(dev_net(ifa->idev->dev), net)) -				return ifa; +		/* prepare for next bucket */ +		state->offset = 0; +		p = 0;  	}  	return NULL;  } @@ -3025,38 +3438,33 @@ static struct inet6_ifaddr *if6_get_next(struct seq_file *seq,  {  	struct if6_iter_state *state = seq->private;  	struct net *net = seq_file_net(seq); -	struct hlist_node *n = &ifa->addr_lst; -	hlist_for_each_entry_continue_rcu_bh(ifa, n, addr_lst) -		if (net_eq(dev_net(ifa->idev->dev), net)) -			return ifa; +	hlist_for_each_entry_continue_rcu_bh(ifa, addr_lst) { +		if (!net_eq(dev_net(ifa->idev->dev), net)) +			continue; +		state->offset++; +		return ifa; +	}  	while (++state->bucket < IN6_ADDR_HSIZE) { -		hlist_for_each_entry_rcu_bh(ifa, n, +		state->offset = 0; +		hlist_for_each_entry_rcu_bh(ifa,  				     &inet6_addr_lst[state->bucket], addr_lst) { -			if (net_eq(dev_net(ifa->idev->dev), net)) -				return ifa; +			if (!net_eq(dev_net(ifa->idev->dev), net)) +				continue; +			state->offset++; +			return ifa;  		}  	}  	return NULL;  } -static struct inet6_ifaddr *if6_get_idx(struct seq_file *seq, loff_t pos) -{ -	struct inet6_ifaddr *ifa = if6_get_first(seq); - -	if (ifa) -		while (pos && (ifa = if6_get_next(seq, ifa)) != NULL) -			--pos; -	return pos ? NULL : ifa; -} -  static void *if6_seq_start(struct seq_file *seq, loff_t *pos)  	__acquires(rcu_bh)  {  	rcu_read_lock_bh(); -	return if6_get_idx(seq, *pos); +	return if6_get_first(seq, *pos);  }  static void *if6_seq_next(struct seq_file *seq, void *v, loff_t *pos) @@ -3082,7 +3490,7 @@ static int if6_seq_show(struct seq_file *seq, void *v)  		   ifp->idev->dev->ifindex,  		   ifp->prefix_len,  		   ifp->scope, -		   ifp->flags, +		   (u8) ifp->flags,  		   ifp->idev->dev->name);  	return 0;  } @@ -3110,14 +3518,14 @@ static const struct file_operations if6_fops = {  static int __net_init if6_proc_net_init(struct net *net)  { -	if (!proc_net_fops_create(net, "if_inet6", S_IRUGO, &if6_fops)) +	if (!proc_create("if_inet6", S_IRUGO, net->proc_net, &if6_fops))  		return -ENOMEM;  	return 0;  }  static void __net_exit if6_proc_net_exit(struct net *net)  { -       proc_net_remove(net, "if_inet6"); +	remove_proc_entry("if_inet6", net->proc_net);  }  static struct pernet_operations if6_proc_net_ops = { @@ -3136,17 +3544,16 @@ void if6_proc_exit(void)  }  #endif	/* CONFIG_PROC_FS */ -#if defined(CONFIG_IPV6_MIP6) || defined(CONFIG_IPV6_MIP6_MODULE) +#if IS_ENABLED(CONFIG_IPV6_MIP6)  /* Check if address is a home address configured on any interface. */ -int ipv6_chk_home_addr(struct net *net, struct in6_addr *addr) +int ipv6_chk_home_addr(struct net *net, const struct in6_addr *addr)  {  	int ret = 0;  	struct inet6_ifaddr *ifp = NULL; -	struct hlist_node *n; -	unsigned int hash = ipv6_addr_hash(addr); +	unsigned int hash = inet6_addr_hash(addr);  	rcu_read_lock_bh(); -	hlist_for_each_entry_rcu_bh(ifp, n, &inet6_addr_lst[hash], addr_lst) { +	hlist_for_each_entry_rcu_bh(ifp, &inet6_addr_lst[hash], addr_lst) {  		if (!net_eq(dev_net(ifp->idev->dev), net))  			continue;  		if (ipv6_addr_equal(&ifp->addr, addr) && @@ -3164,27 +3571,31 @@ int ipv6_chk_home_addr(struct net *net, struct in6_addr *addr)   *	Periodic address status verification   */ -static void addrconf_verify(unsigned long foo) +static void addrconf_verify_rtnl(void)  {  	unsigned long now, next, next_sec, next_sched;  	struct inet6_ifaddr *ifp; -	struct hlist_node *node;  	int i; +	ASSERT_RTNL(); +  	rcu_read_lock_bh(); -	spin_lock(&addrconf_verify_lock);  	now = jiffies;  	next = round_jiffies_up(now + ADDR_CHECK_FREQUENCY); -	del_timer(&addr_chk_timer); +	cancel_delayed_work(&addr_chk_work);  	for (i = 0; i < IN6_ADDR_HSIZE; i++) {  restart: -		hlist_for_each_entry_rcu_bh(ifp, node, -					 &inet6_addr_lst[i], addr_lst) { +		hlist_for_each_entry_rcu_bh(ifp, &inet6_addr_lst[i], addr_lst) {  			unsigned long age; -			if (ifp->flags & IFA_F_PERMANENT) +			/* When setting preferred_lft to a value not zero or +			 * infinity, while valid_lft is infinity +			 * IFA_F_PERMANENT has a non-infinity life time. +			 */ +			if ((ifp->flags & IFA_F_PERMANENT) && +			    (ifp->prefered_lft == INFINITY_LIFE_TIME))  				continue;  			spin_lock(&ifp->lock); @@ -3209,7 +3620,8 @@ restart:  					ifp->flags |= IFA_F_DEPRECATED;  				} -				if (time_before(ifp->tstamp + ifp->valid_lft * HZ, next)) +				if ((ifp->valid_lft != INFINITY_LIFE_TIME) && +				    (time_before(ifp->tstamp + ifp->valid_lft * HZ, next)))  					next = ifp->tstamp + ifp->valid_lft * HZ;  				spin_unlock(&ifp->lock); @@ -3221,12 +3633,11 @@ restart:  					in6_ifa_put(ifp);  					goto restart;  				} -#ifdef CONFIG_IPV6_PRIVACY  			} else if ((ifp->flags&IFA_F_TEMPORARY) &&  				   !(ifp->flags&IFA_F_TENTATIVE)) {  				unsigned long regen_advance = ifp->idev->cnf.regen_max_retry *  					ifp->idev->cnf.dad_transmits * -					ifp->idev->nd_parms->retrans_time / HZ; +					NEIGH_VAR(ifp->idev->nd_parms, RETRANS_TIME) / HZ;  				if (age >= ifp->prefered_lft - regen_advance) {  					struct inet6_ifaddr *ifpub = ifp->ifpub; @@ -3249,7 +3660,6 @@ restart:  				} else if (time_before(ifp->tstamp + ifp->prefered_lft * HZ - regen_advance * HZ, next))  					next = ifp->tstamp + ifp->prefered_lft * HZ - regen_advance * HZ;  				spin_unlock(&ifp->lock); -#endif  			} else {  				/* ifp->prefered_lft <= ifp->valid_lft */  				if (time_before(ifp->tstamp + ifp->prefered_lft * HZ, next)) @@ -3270,27 +3680,38 @@ restart:  	if (time_before(next_sched, jiffies + ADDRCONF_TIMER_FUZZ_MAX))  		next_sched = jiffies + ADDRCONF_TIMER_FUZZ_MAX; -	ADBG((KERN_DEBUG "now = %lu, schedule = %lu, rounded schedule = %lu => %lu\n", -	      now, next, next_sec, next_sched)); - -	addr_chk_timer.expires = next_sched; -	add_timer(&addr_chk_timer); -	spin_unlock(&addrconf_verify_lock); +	ADBG(KERN_DEBUG "now = %lu, schedule = %lu, rounded schedule = %lu => %lu\n", +	      now, next, next_sec, next_sched); +	mod_delayed_work(addrconf_wq, &addr_chk_work, next_sched - now);  	rcu_read_unlock_bh();  } -static struct in6_addr *extract_addr(struct nlattr *addr, struct nlattr *local) +static void addrconf_verify_work(struct work_struct *w) +{ +	rtnl_lock(); +	addrconf_verify_rtnl(); +	rtnl_unlock(); +} + +static void addrconf_verify(void) +{ +	mod_delayed_work(addrconf_wq, &addr_chk_work, 0); +} + +static struct in6_addr *extract_addr(struct nlattr *addr, struct nlattr *local, +				     struct in6_addr **peer_pfx)  {  	struct in6_addr *pfx = NULL; +	*peer_pfx = NULL; +  	if (addr)  		pfx = nla_data(addr);  	if (local) {  		if (pfx && nla_memcmp(local, pfx, sizeof(*pfx))) -			pfx = NULL; -		else -			pfx = nla_data(local); +			*peer_pfx = pfx; +		pfx = nla_data(local);  	}  	return pfx; @@ -3300,15 +3721,17 @@ static const struct nla_policy ifa_ipv6_policy[IFA_MAX+1] = {  	[IFA_ADDRESS]		= { .len = sizeof(struct in6_addr) },  	[IFA_LOCAL]		= { .len = sizeof(struct in6_addr) },  	[IFA_CACHEINFO]		= { .len = sizeof(struct ifa_cacheinfo) }, +	[IFA_FLAGS]		= { .len = sizeof(u32) },  };  static int -inet6_rtm_deladdr(struct sk_buff *skb, struct nlmsghdr *nlh, void *arg) +inet6_rtm_deladdr(struct sk_buff *skb, struct nlmsghdr *nlh)  {  	struct net *net = sock_net(skb->sk);  	struct ifaddrmsg *ifm;  	struct nlattr *tb[IFA_MAX+1]; -	struct in6_addr *pfx; +	struct in6_addr *pfx, *peer_pfx; +	u32 ifa_flags;  	int err;  	err = nlmsg_parse(nlh, sizeof(*ifm), tb, IFA_MAX, ifa_ipv6_policy); @@ -3316,23 +3739,37 @@ inet6_rtm_deladdr(struct sk_buff *skb, struct nlmsghdr *nlh, void *arg)  		return err;  	ifm = nlmsg_data(nlh); -	pfx = extract_addr(tb[IFA_ADDRESS], tb[IFA_LOCAL]); +	pfx = extract_addr(tb[IFA_ADDRESS], tb[IFA_LOCAL], &peer_pfx);  	if (pfx == NULL)  		return -EINVAL; -	return inet6_addr_del(net, ifm->ifa_index, pfx, ifm->ifa_prefixlen); +	ifa_flags = tb[IFA_FLAGS] ? nla_get_u32(tb[IFA_FLAGS]) : ifm->ifa_flags; + +	/* We ignore other flags so far. */ +	ifa_flags &= IFA_F_MANAGETEMPADDR; + +	return inet6_addr_del(net, ifm->ifa_index, ifa_flags, pfx, +			      ifm->ifa_prefixlen);  } -static int inet6_addr_modify(struct inet6_ifaddr *ifp, u8 ifa_flags, +static int inet6_addr_modify(struct inet6_ifaddr *ifp, u32 ifa_flags,  			     u32 prefered_lft, u32 valid_lft)  {  	u32 flags;  	clock_t expires;  	unsigned long timeout; +	bool was_managetempaddr; +	bool had_prefixroute; + +	ASSERT_RTNL();  	if (!valid_lft || (prefered_lft > valid_lft))  		return -EINVAL; +	if (ifa_flags & IFA_F_MANAGETEMPADDR && +	    (ifp->flags & IFA_F_TEMPORARY || ifp->prefix_len != 64)) +		return -EINVAL; +  	timeout = addrconf_timeout_fixup(valid_lft, HZ);  	if (addrconf_finite_timeout(timeout)) {  		expires = jiffies_to_clock_t(timeout * HZ); @@ -3352,7 +3789,13 @@ static int inet6_addr_modify(struct inet6_ifaddr *ifp, u8 ifa_flags,  	}  	spin_lock_bh(&ifp->lock); -	ifp->flags = (ifp->flags & ~(IFA_F_DEPRECATED | IFA_F_PERMANENT | IFA_F_NODAD | IFA_F_HOMEADDRESS)) | ifa_flags; +	was_managetempaddr = ifp->flags & IFA_F_MANAGETEMPADDR; +	had_prefixroute = ifp->flags & IFA_F_PERMANENT && +			  !(ifp->flags & IFA_F_NOPREFIXROUTE); +	ifp->flags &= ~(IFA_F_DEPRECATED | IFA_F_PERMANENT | IFA_F_NODAD | +			IFA_F_HOMEADDRESS | IFA_F_MANAGETEMPADDR | +			IFA_F_NOPREFIXROUTE); +	ifp->flags |= ifa_flags;  	ifp->tstamp = jiffies;  	ifp->valid_lft = valid_lft;  	ifp->prefered_lft = prefered_lft; @@ -3361,24 +3804,46 @@ static int inet6_addr_modify(struct inet6_ifaddr *ifp, u8 ifa_flags,  	if (!(ifp->flags&IFA_F_TENTATIVE))  		ipv6_ifa_notify(0, ifp); -	addrconf_prefix_route(&ifp->addr, ifp->prefix_len, ifp->idev->dev, -			      expires, flags); -	addrconf_verify(0); +	if (!(ifa_flags & IFA_F_NOPREFIXROUTE)) { +		addrconf_prefix_route(&ifp->addr, ifp->prefix_len, ifp->idev->dev, +				      expires, flags); +	} else if (had_prefixroute) { +		enum cleanup_prefix_rt_t action; +		unsigned long rt_expires; + +		write_lock_bh(&ifp->idev->lock); +		action = check_cleanup_prefix_route(ifp, &rt_expires); +		write_unlock_bh(&ifp->idev->lock); + +		if (action != CLEANUP_PREFIX_RT_NOP) { +			cleanup_prefix_route(ifp, rt_expires, +				action == CLEANUP_PREFIX_RT_DEL); +		} +	} + +	if (was_managetempaddr || ifp->flags & IFA_F_MANAGETEMPADDR) { +		if (was_managetempaddr && !(ifp->flags & IFA_F_MANAGETEMPADDR)) +			valid_lft = prefered_lft = 0; +		manage_tempaddrs(ifp->idev, ifp, valid_lft, prefered_lft, +				 !was_managetempaddr, jiffies); +	} + +	addrconf_verify_rtnl();  	return 0;  }  static int -inet6_rtm_newaddr(struct sk_buff *skb, struct nlmsghdr *nlh, void *arg) +inet6_rtm_newaddr(struct sk_buff *skb, struct nlmsghdr *nlh)  {  	struct net *net = sock_net(skb->sk);  	struct ifaddrmsg *ifm;  	struct nlattr *tb[IFA_MAX+1]; -	struct in6_addr *pfx; +	struct in6_addr *pfx, *peer_pfx;  	struct inet6_ifaddr *ifa;  	struct net_device *dev;  	u32 valid_lft = INFINITY_LIFE_TIME, preferred_lft = INFINITY_LIFE_TIME; -	u8 ifa_flags; +	u32 ifa_flags;  	int err;  	err = nlmsg_parse(nlh, sizeof(*ifm), tb, IFA_MAX, ifa_ipv6_policy); @@ -3386,7 +3851,7 @@ inet6_rtm_newaddr(struct sk_buff *skb, struct nlmsghdr *nlh, void *arg)  		return err;  	ifm = nlmsg_data(nlh); -	pfx = extract_addr(tb[IFA_ADDRESS], tb[IFA_LOCAL]); +	pfx = extract_addr(tb[IFA_ADDRESS], tb[IFA_LOCAL], &peer_pfx);  	if (pfx == NULL)  		return -EINVAL; @@ -3405,16 +3870,19 @@ inet6_rtm_newaddr(struct sk_buff *skb, struct nlmsghdr *nlh, void *arg)  	if (dev == NULL)  		return -ENODEV; +	ifa_flags = tb[IFA_FLAGS] ? nla_get_u32(tb[IFA_FLAGS]) : ifm->ifa_flags; +  	/* We ignore other flags so far. */ -	ifa_flags = ifm->ifa_flags & (IFA_F_NODAD | IFA_F_HOMEADDRESS); +	ifa_flags &= IFA_F_NODAD | IFA_F_HOMEADDRESS | IFA_F_MANAGETEMPADDR | +		     IFA_F_NOPREFIXROUTE;  	ifa = ipv6_get_ifaddr(net, pfx, dev, 1);  	if (ifa == NULL) {  		/*  		 * It would be best to check for !NLM_F_CREATE here but -		 * userspace alreay relies on not having to provide this. +		 * userspace already relies on not having to provide this.  		 */ -		return inet6_addr_add(net, ifm->ifa_index, pfx, +		return inet6_addr_add(net, ifm->ifa_index, pfx, peer_pfx,  				      ifm->ifa_prefixlen, ifa_flags,  				      preferred_lft, valid_lft);  	} @@ -3430,7 +3898,7 @@ inet6_rtm_newaddr(struct sk_buff *skb, struct nlmsghdr *nlh, void *arg)  	return err;  } -static void put_ifaddrmsg(struct nlmsghdr *nlh, u8 prefixlen, u8 flags, +static void put_ifaddrmsg(struct nlmsghdr *nlh, u8 prefixlen, u32 flags,  			  u8 scope, int ifindex)  {  	struct ifaddrmsg *ifm; @@ -3471,24 +3939,27 @@ static inline int rt_scope(int ifa_scope)  static inline int inet6_ifaddr_msgsize(void)  {  	return NLMSG_ALIGN(sizeof(struct ifaddrmsg)) +	       + nla_total_size(16) /* IFA_LOCAL */  	       + nla_total_size(16) /* IFA_ADDRESS */ -	       + nla_total_size(sizeof(struct ifa_cacheinfo)); +	       + nla_total_size(sizeof(struct ifa_cacheinfo)) +	       + nla_total_size(4)  /* IFA_FLAGS */;  }  static int inet6_fill_ifaddr(struct sk_buff *skb, struct inet6_ifaddr *ifa, -			     u32 pid, u32 seq, int event, unsigned int flags) +			     u32 portid, u32 seq, int event, unsigned int flags)  {  	struct nlmsghdr  *nlh;  	u32 preferred, valid; -	nlh = nlmsg_put(skb, pid, seq, event, sizeof(struct ifaddrmsg), flags); +	nlh = nlmsg_put(skb, portid, seq, event, sizeof(struct ifaddrmsg), flags);  	if (nlh == NULL)  		return -EMSGSIZE;  	put_ifaddrmsg(nlh, ifa->prefix_len, ifa->flags, rt_scope(ifa->scope),  		      ifa->idev->dev->ifindex); -	if (!(ifa->flags&IFA_F_PERMANENT)) { +	if (!((ifa->flags&IFA_F_PERMANENT) && +	      (ifa->prefered_lft == INFINITY_LIFE_TIME))) {  		preferred = ifa->prefered_lft;  		valid = ifa->valid_lft;  		if (preferred != INFINITY_LIFE_TIME) { @@ -3509,17 +3980,29 @@ static int inet6_fill_ifaddr(struct sk_buff *skb, struct inet6_ifaddr *ifa,  		valid = INFINITY_LIFE_TIME;  	} -	if (nla_put(skb, IFA_ADDRESS, 16, &ifa->addr) < 0 || -	    put_cacheinfo(skb, ifa->cstamp, ifa->tstamp, preferred, valid) < 0) { -		nlmsg_cancel(skb, nlh); -		return -EMSGSIZE; -	} +	if (!ipv6_addr_any(&ifa->peer_addr)) { +		if (nla_put(skb, IFA_LOCAL, 16, &ifa->addr) < 0 || +		    nla_put(skb, IFA_ADDRESS, 16, &ifa->peer_addr) < 0) +			goto error; +	} else +		if (nla_put(skb, IFA_ADDRESS, 16, &ifa->addr) < 0) +			goto error; + +	if (put_cacheinfo(skb, ifa->cstamp, ifa->tstamp, preferred, valid) < 0) +		goto error; + +	if (nla_put_u32(skb, IFA_FLAGS, ifa->flags) < 0) +		goto error;  	return nlmsg_end(skb, nlh); + +error: +	nlmsg_cancel(skb, nlh); +	return -EMSGSIZE;  }  static int inet6_fill_ifmcaddr(struct sk_buff *skb, struct ifmcaddr6 *ifmca, -				u32 pid, u32 seq, int event, u16 flags) +				u32 portid, u32 seq, int event, u16 flags)  {  	struct nlmsghdr  *nlh;  	u8 scope = RT_SCOPE_UNIVERSE; @@ -3528,7 +4011,7 @@ static int inet6_fill_ifmcaddr(struct sk_buff *skb, struct ifmcaddr6 *ifmca,  	if (ipv6_addr_scope(&ifmca->mca_addr) & IFA_SITE)  		scope = RT_SCOPE_SITE; -	nlh = nlmsg_put(skb, pid, seq, event, sizeof(struct ifaddrmsg), flags); +	nlh = nlmsg_put(skb, portid, seq, event, sizeof(struct ifaddrmsg), flags);  	if (nlh == NULL)  		return -EMSGSIZE; @@ -3544,7 +4027,7 @@ static int inet6_fill_ifmcaddr(struct sk_buff *skb, struct ifmcaddr6 *ifmca,  }  static int inet6_fill_ifacaddr(struct sk_buff *skb, struct ifacaddr6 *ifaca, -				u32 pid, u32 seq, int event, unsigned int flags) +				u32 portid, u32 seq, int event, unsigned int flags)  {  	struct nlmsghdr  *nlh;  	u8 scope = RT_SCOPE_UNIVERSE; @@ -3553,7 +4036,7 @@ static int inet6_fill_ifacaddr(struct sk_buff *skb, struct ifacaddr6 *ifaca,  	if (ipv6_addr_scope(&ifaca->aca_addr) & IFA_SITE)  		scope = RT_SCOPE_SITE; -	nlh = nlmsg_put(skb, pid, seq, event, sizeof(struct ifaddrmsg), flags); +	nlh = nlmsg_put(skb, portid, seq, event, sizeof(struct ifaddrmsg), flags);  	if (nlh == NULL)  		return -EMSGSIZE; @@ -3594,12 +4077,13 @@ static int in6_dump_addrs(struct inet6_dev *idev, struct sk_buff *skb,  			if (++ip_idx < s_ip_idx)  				continue;  			err = inet6_fill_ifaddr(skb, ifa, -						NETLINK_CB(cb->skb).pid, +						NETLINK_CB(cb->skb).portid,  						cb->nlh->nlmsg_seq,  						RTM_NEWADDR,  						NLM_F_MULTI);  			if (err <= 0)  				break; +			nl_dump_check_consistent(cb, nlmsg_hdr(skb));  		}  		break;  	} @@ -3610,7 +4094,7 @@ static int in6_dump_addrs(struct inet6_dev *idev, struct sk_buff *skb,  			if (ip_idx < s_ip_idx)  				continue;  			err = inet6_fill_ifmcaddr(skb, ifmca, -						  NETLINK_CB(cb->skb).pid, +						  NETLINK_CB(cb->skb).portid,  						  cb->nlh->nlmsg_seq,  						  RTM_GETMULTICAST,  						  NLM_F_MULTI); @@ -3625,7 +4109,7 @@ static int in6_dump_addrs(struct inet6_dev *idev, struct sk_buff *skb,  			if (ip_idx < s_ip_idx)  				continue;  			err = inet6_fill_ifacaddr(skb, ifaca, -						  NETLINK_CB(cb->skb).pid, +						  NETLINK_CB(cb->skb).portid,  						  cb->nlh->nlmsg_seq,  						  RTM_GETANYCAST,  						  NLM_F_MULTI); @@ -3651,17 +4135,17 @@ static int inet6_dump_addr(struct sk_buff *skb, struct netlink_callback *cb,  	struct net_device *dev;  	struct inet6_dev *idev;  	struct hlist_head *head; -	struct hlist_node *node;  	s_h = cb->args[0];  	s_idx = idx = cb->args[1];  	s_ip_idx = ip_idx = cb->args[2];  	rcu_read_lock(); +	cb->seq = atomic_read(&net->ipv6.dev_addr_genid) ^ net->dev_base_seq;  	for (h = s_h; h < NETDEV_HASHENTRIES; h++, s_idx = 0) {  		idx = 0;  		head = &net->dev_index_head[h]; -		hlist_for_each_entry_rcu(dev, node, head, index_hlist) { +		hlist_for_each_entry_rcu(dev, head, index_hlist) {  			if (idx < s_idx)  				goto cont;  			if (h > s_h || idx > s_idx) @@ -3709,13 +4193,12 @@ static int inet6_dump_ifacaddr(struct sk_buff *skb, struct netlink_callback *cb)  	return inet6_dump_addr(skb, cb, type);  } -static int inet6_rtm_getaddr(struct sk_buff *in_skb, struct nlmsghdr* nlh, -			     void *arg) +static int inet6_rtm_getaddr(struct sk_buff *in_skb, struct nlmsghdr *nlh)  {  	struct net *net = sock_net(in_skb->sk);  	struct ifaddrmsg *ifm;  	struct nlattr *tb[IFA_MAX+1]; -	struct in6_addr *addr = NULL; +	struct in6_addr *addr = NULL, *peer;  	struct net_device *dev = NULL;  	struct inet6_ifaddr *ifa;  	struct sk_buff *skb; @@ -3725,7 +4208,7 @@ static int inet6_rtm_getaddr(struct sk_buff *in_skb, struct nlmsghdr* nlh,  	if (err < 0)  		goto errout; -	addr = extract_addr(tb[IFA_ADDRESS], tb[IFA_LOCAL]); +	addr = extract_addr(tb[IFA_ADDRESS], tb[IFA_LOCAL], &peer);  	if (addr == NULL) {  		err = -EINVAL;  		goto errout; @@ -3747,7 +4230,7 @@ static int inet6_rtm_getaddr(struct sk_buff *in_skb, struct nlmsghdr* nlh,  		goto errout_ifa;  	} -	err = inet6_fill_ifaddr(skb, ifa, NETLINK_CB(in_skb).pid, +	err = inet6_fill_ifaddr(skb, ifa, NETLINK_CB(in_skb).portid,  				nlh->nlmsg_seq, RTM_NEWADDR, 0);  	if (err < 0) {  		/* -EMSGSIZE implies BUG in inet6_ifaddr_msgsize() */ @@ -3755,7 +4238,7 @@ static int inet6_rtm_getaddr(struct sk_buff *in_skb, struct nlmsghdr* nlh,  		kfree_skb(skb);  		goto errout_ifa;  	} -	err = rtnl_unicast(skb, net, NETLINK_CB(in_skb).pid); +	err = rtnl_unicast(skb, net, NETLINK_CB(in_skb).portid);  errout_ifa:  	in6_ifa_put(ifa);  errout: @@ -3805,13 +4288,15 @@ static inline void ipv6_store_devconf(struct ipv6_devconf *cnf,  	array[DEVCONF_RTR_SOLICIT_DELAY] =  		jiffies_to_msecs(cnf->rtr_solicit_delay);  	array[DEVCONF_FORCE_MLD_VERSION] = cnf->force_mld_version; -#ifdef CONFIG_IPV6_PRIVACY +	array[DEVCONF_MLDV1_UNSOLICITED_REPORT_INTERVAL] = +		jiffies_to_msecs(cnf->mldv1_unsolicited_report_interval); +	array[DEVCONF_MLDV2_UNSOLICITED_REPORT_INTERVAL] = +		jiffies_to_msecs(cnf->mldv2_unsolicited_report_interval);  	array[DEVCONF_USE_TEMPADDR] = cnf->use_tempaddr;  	array[DEVCONF_TEMP_VALID_LFT] = cnf->temp_valid_lft;  	array[DEVCONF_TEMP_PREFERED_LFT] = cnf->temp_prefered_lft;  	array[DEVCONF_REGEN_MAX_RETRY] = cnf->regen_max_retry;  	array[DEVCONF_MAX_DESYNC_FACTOR] = cnf->max_desync_factor; -#endif  	array[DEVCONF_MAX_ADDRESSES] = cnf->max_addresses;  	array[DEVCONF_ACCEPT_RA_DEFRTR] = cnf->accept_ra_defrtr;  	array[DEVCONF_ACCEPT_RA_PINFO] = cnf->accept_ra_pinfo; @@ -3834,6 +4319,8 @@ static inline void ipv6_store_devconf(struct ipv6_devconf *cnf,  	array[DEVCONF_DISABLE_IPV6] = cnf->disable_ipv6;  	array[DEVCONF_ACCEPT_DAD] = cnf->accept_dad;  	array[DEVCONF_FORCE_TLLAO] = cnf->force_tllao; +	array[DEVCONF_NDISC_NOTIFY] = cnf->ndisc_notify; +	array[DEVCONF_SUPPRESS_FRAG_NDISC] = cnf->suppress_frag_ndisc;  }  static inline size_t inet6_ifla6_size(void) @@ -3842,7 +4329,8 @@ static inline size_t inet6_ifla6_size(void)  	     + nla_total_size(sizeof(struct ifla_cacheinfo))  	     + nla_total_size(DEVCONF_MAX * 4) /* IFLA_INET6_CONF */  	     + nla_total_size(IPSTATS_MIB_MAX * 8) /* IFLA_INET6_STATS */ -	     + nla_total_size(ICMP6_MIB_MAX * 8); /* IFLA_INET6_ICMP6STATS */ +	     + nla_total_size(ICMP6_MIB_MAX * 8) /* IFLA_INET6_ICMP6STATS */ +	     + nla_total_size(sizeof(struct in6_addr)); /* IFLA_INET6_TOKEN */  }  static inline size_t inet6_if_nlmsg_size(void) @@ -3855,7 +4343,7 @@ static inline size_t inet6_if_nlmsg_size(void)  	       + nla_total_size(inet6_ifla6_size()); /* IFLA_PROTINFO */  } -static inline void __snmp6_fill_stats(u64 *stats, void __percpu **mib, +static inline void __snmp6_fill_statsdev(u64 *stats, atomic_long_t *mib,  				      int items, int bytes)  {  	int i; @@ -3865,12 +4353,12 @@ static inline void __snmp6_fill_stats(u64 *stats, void __percpu **mib,  	/* Use put_unaligned() because stats may not be aligned for u64. */  	put_unaligned(items, &stats[0]);  	for (i = 1; i < items; i++) -		put_unaligned(snmp_fold_field(mib, i), &stats[i]); +		put_unaligned(atomic_long_read(&mib[i]), &stats[i]);  	memset(&stats[items], 0, pad);  } -static inline void __snmp6_fill_stats64(u64 *stats, void __percpu **mib, +static inline void __snmp6_fill_stats64(u64 *stats, void __percpu *mib,  				      int items, int bytes, size_t syncpoff)  {  	int i; @@ -3890,11 +4378,11 @@ static void snmp6_fill_stats(u64 *stats, struct inet6_dev *idev, int attrtype,  {  	switch (attrtype) {  	case IFLA_INET6_STATS: -		__snmp6_fill_stats64(stats, (void __percpu **)idev->stats.ipv6, +		__snmp6_fill_stats64(stats, idev->stats.ipv6,  				     IPSTATS_MIB_MAX, bytes, offsetof(struct ipstats_mib, syncp));  		break;  	case IFLA_INET6_ICMP6STATS: -		__snmp6_fill_stats(stats, (void __percpu **)idev->stats.icmpv6, ICMP6_MIB_MAX, bytes); +		__snmp6_fill_statsdev(stats, idev->stats.icmpv6dev->mibs, ICMP6_MIB_MAX, bytes);  		break;  	}  } @@ -3904,14 +4392,14 @@ static int inet6_fill_ifla6_attrs(struct sk_buff *skb, struct inet6_dev *idev)  	struct nlattr *nla;  	struct ifla_cacheinfo ci; -	NLA_PUT_U32(skb, IFLA_INET6_FLAGS, idev->if_flags); - +	if (nla_put_u32(skb, IFLA_INET6_FLAGS, idev->if_flags)) +		goto nla_put_failure;  	ci.max_reasm_len = IPV6_MAXPLEN;  	ci.tstamp = cstamp_delta(idev->tstamp);  	ci.reachable_time = jiffies_to_msecs(idev->nd_parms->reachable_time); -	ci.retrans_time = jiffies_to_msecs(idev->nd_parms->retrans_time); -	NLA_PUT(skb, IFLA_INET6_CACHEINFO, sizeof(ci), &ci); - +	ci.retrans_time = jiffies_to_msecs(NEIGH_VAR(idev->nd_parms, RETRANS_TIME)); +	if (nla_put(skb, IFLA_INET6_CACHEINFO, sizeof(ci), &ci)) +		goto nla_put_failure;  	nla = nla_reserve(skb, IFLA_INET6_CONF, DEVCONF_MAX * sizeof(s32));  	if (nla == NULL)  		goto nla_put_failure; @@ -3929,6 +4417,13 @@ static int inet6_fill_ifla6_attrs(struct sk_buff *skb, struct inet6_dev *idev)  		goto nla_put_failure;  	snmp6_fill_stats(nla_data(nla), idev, IFLA_INET6_ICMP6STATS, nla_len(nla)); +	nla = nla_reserve(skb, IFLA_INET6_TOKEN, sizeof(struct in6_addr)); +	if (nla == NULL) +		goto nla_put_failure; +	read_lock_bh(&idev->lock); +	memcpy(nla_data(nla), idev->token.s6_addr, nla_len(nla)); +	read_unlock_bh(&idev->lock); +  	return 0;  nla_put_failure: @@ -3956,15 +4451,94 @@ static int inet6_fill_link_af(struct sk_buff *skb, const struct net_device *dev)  	return 0;  } +static int inet6_set_iftoken(struct inet6_dev *idev, struct in6_addr *token) +{ +	struct inet6_ifaddr *ifp; +	struct net_device *dev = idev->dev; +	bool update_rs = false; +	struct in6_addr ll_addr; + +	ASSERT_RTNL(); + +	if (token == NULL) +		return -EINVAL; +	if (ipv6_addr_any(token)) +		return -EINVAL; +	if (dev->flags & (IFF_LOOPBACK | IFF_NOARP)) +		return -EINVAL; +	if (!ipv6_accept_ra(idev)) +		return -EINVAL; +	if (idev->cnf.rtr_solicits <= 0) +		return -EINVAL; + +	write_lock_bh(&idev->lock); + +	BUILD_BUG_ON(sizeof(token->s6_addr) != 16); +	memcpy(idev->token.s6_addr + 8, token->s6_addr + 8, 8); + +	write_unlock_bh(&idev->lock); + +	if (!idev->dead && (idev->if_flags & IF_READY) && +	    !ipv6_get_lladdr(dev, &ll_addr, IFA_F_TENTATIVE | +			     IFA_F_OPTIMISTIC)) { + +		/* If we're not ready, then normal ifup will take care +		 * of this. Otherwise, we need to request our rs here. +		 */ +		ndisc_send_rs(dev, &ll_addr, &in6addr_linklocal_allrouters); +		update_rs = true; +	} + +	write_lock_bh(&idev->lock); + +	if (update_rs) { +		idev->if_flags |= IF_RS_SENT; +		idev->rs_probes = 1; +		addrconf_mod_rs_timer(idev, idev->cnf.rtr_solicit_interval); +	} + +	/* Well, that's kinda nasty ... */ +	list_for_each_entry(ifp, &idev->addr_list, if_list) { +		spin_lock(&ifp->lock); +		if (ifp->tokenized) { +			ifp->valid_lft = 0; +			ifp->prefered_lft = 0; +		} +		spin_unlock(&ifp->lock); +	} + +	write_unlock_bh(&idev->lock); +	addrconf_verify_rtnl(); +	return 0; +} + +static int inet6_set_link_af(struct net_device *dev, const struct nlattr *nla) +{ +	int err = -EINVAL; +	struct inet6_dev *idev = __in6_dev_get(dev); +	struct nlattr *tb[IFLA_INET6_MAX + 1]; + +	if (!idev) +		return -EAFNOSUPPORT; + +	if (nla_parse_nested(tb, IFLA_INET6_MAX, nla, NULL) < 0) +		BUG(); + +	if (tb[IFLA_INET6_TOKEN]) +		err = inet6_set_iftoken(idev, nla_data(tb[IFLA_INET6_TOKEN])); + +	return err; +} +  static int inet6_fill_ifinfo(struct sk_buff *skb, struct inet6_dev *idev, -			     u32 pid, u32 seq, int event, unsigned int flags) +			     u32 portid, u32 seq, int event, unsigned int flags)  {  	struct net_device *dev = idev->dev;  	struct ifinfomsg *hdr;  	struct nlmsghdr *nlh;  	void *protoinfo; -	nlh = nlmsg_put(skb, pid, seq, event, sizeof(*hdr), flags); +	nlh = nlmsg_put(skb, portid, seq, event, sizeof(*hdr), flags);  	if (nlh == NULL)  		return -EMSGSIZE; @@ -3976,15 +4550,13 @@ static int inet6_fill_ifinfo(struct sk_buff *skb, struct inet6_dev *idev,  	hdr->ifi_flags = dev_get_flags(dev);  	hdr->ifi_change = 0; -	NLA_PUT_STRING(skb, IFLA_IFNAME, dev->name); - -	if (dev->addr_len) -		NLA_PUT(skb, IFLA_ADDRESS, dev->addr_len, dev->dev_addr); - -	NLA_PUT_U32(skb, IFLA_MTU, dev->mtu); -	if (dev->ifindex != dev->iflink) -		NLA_PUT_U32(skb, IFLA_LINK, dev->iflink); - +	if (nla_put_string(skb, IFLA_IFNAME, dev->name) || +	    (dev->addr_len && +	     nla_put(skb, IFLA_ADDRESS, dev->addr_len, dev->dev_addr)) || +	    nla_put_u32(skb, IFLA_MTU, dev->mtu) || +	    (dev->ifindex != dev->iflink && +	     nla_put_u32(skb, IFLA_LINK, dev->iflink))) +		goto nla_put_failure;  	protoinfo = nla_nest_start(skb, IFLA_PROTINFO);  	if (protoinfo == NULL)  		goto nla_put_failure; @@ -4008,7 +4580,6 @@ static int inet6_dump_ifinfo(struct sk_buff *skb, struct netlink_callback *cb)  	struct net_device *dev;  	struct inet6_dev *idev;  	struct hlist_head *head; -	struct hlist_node *node;  	s_h = cb->args[0];  	s_idx = cb->args[1]; @@ -4017,14 +4588,14 @@ static int inet6_dump_ifinfo(struct sk_buff *skb, struct netlink_callback *cb)  	for (h = s_h; h < NETDEV_HASHENTRIES; h++, s_idx = 0) {  		idx = 0;  		head = &net->dev_index_head[h]; -		hlist_for_each_entry_rcu(dev, node, head, index_hlist) { +		hlist_for_each_entry_rcu(dev, head, index_hlist) {  			if (idx < s_idx)  				goto cont;  			idev = __in6_dev_get(dev);  			if (!idev)  				goto cont;  			if (inet6_fill_ifinfo(skb, idev, -					      NETLINK_CB(cb->skb).pid, +					      NETLINK_CB(cb->skb).portid,  					      cb->nlh->nlmsg_seq,  					      RTM_NEWLINK, NLM_F_MULTI) <= 0)  				goto out; @@ -4057,11 +4628,11 @@ void inet6_ifinfo_notify(int event, struct inet6_dev *idev)  		kfree_skb(skb);  		goto errout;  	} -	rtnl_notify(skb, net, 0, RTNLGRP_IPV6_IFADDR, NULL, GFP_ATOMIC); +	rtnl_notify(skb, net, 0, RTNLGRP_IPV6_IFINFO, NULL, GFP_ATOMIC);  	return;  errout:  	if (err < 0) -		rtnl_set_sk_err(net, RTNLGRP_IPV6_IFADDR, err); +		rtnl_set_sk_err(net, RTNLGRP_IPV6_IFINFO, err);  }  static inline size_t inet6_prefix_nlmsg_size(void) @@ -4072,14 +4643,14 @@ static inline size_t inet6_prefix_nlmsg_size(void)  }  static int inet6_fill_prefix(struct sk_buff *skb, struct inet6_dev *idev, -			     struct prefix_info *pinfo, u32 pid, u32 seq, +			     struct prefix_info *pinfo, u32 portid, u32 seq,  			     int event, unsigned int flags)  {  	struct prefixmsg *pmsg;  	struct nlmsghdr *nlh;  	struct prefix_cacheinfo	ci; -	nlh = nlmsg_put(skb, pid, seq, event, sizeof(*pmsg), flags); +	nlh = nlmsg_put(skb, portid, seq, event, sizeof(*pmsg), flags);  	if (nlh == NULL)  		return -EMSGSIZE; @@ -4097,12 +4668,12 @@ static int inet6_fill_prefix(struct sk_buff *skb, struct inet6_dev *idev,  	if (pinfo->autoconf)  		pmsg->prefix_flags |= IF_PREFIX_AUTOCONF; -	NLA_PUT(skb, PREFIX_ADDRESS, sizeof(pinfo->prefix), &pinfo->prefix); - +	if (nla_put(skb, PREFIX_ADDRESS, sizeof(pinfo->prefix), &pinfo->prefix)) +		goto nla_put_failure;  	ci.preferred_time = ntohl(pinfo->prefered);  	ci.valid_time = ntohl(pinfo->valid); -	NLA_PUT(skb, PREFIX_CACHEINFO, sizeof(ci), &ci); - +	if (nla_put(skb, PREFIX_CACHEINFO, sizeof(ci), &ci)) +		goto nla_put_failure;  	return nlmsg_end(skb, nlh);  nla_put_failure: @@ -4137,6 +4708,11 @@ errout:  static void __ipv6_ifa_notify(int event, struct inet6_ifaddr *ifp)  { +	struct net *net = dev_net(ifp->idev->dev); + +	if (event) +		ASSERT_RTNL(); +  	inet6_ifa_notify(event ? : RTM_NEWADDR, ifp);  	switch (event) { @@ -4151,18 +4727,34 @@ static void __ipv6_ifa_notify(int event, struct inet6_ifaddr *ifp)  			ip6_ins_rt(ifp->rt);  		if (ifp->idev->cnf.forwarding)  			addrconf_join_anycast(ifp); +		if (!ipv6_addr_any(&ifp->peer_addr)) +			addrconf_prefix_route(&ifp->peer_addr, 128, +					      ifp->idev->dev, 0, 0);  		break;  	case RTM_DELADDR:  		if (ifp->idev->cnf.forwarding)  			addrconf_leave_anycast(ifp);  		addrconf_leave_solict(ifp->idev, &ifp->addr); +		if (!ipv6_addr_any(&ifp->peer_addr)) { +			struct rt6_info *rt; +			struct net_device *dev = ifp->idev->dev; + +			rt = rt6_lookup(dev_net(dev), &ifp->peer_addr, NULL, +					dev->ifindex, 1); +			if (rt) { +				dst_hold(&rt->dst); +				if (ip6_del_rt(rt)) +					dst_free(&rt->dst); +			} +		}  		dst_hold(&ifp->rt->dst); -		if (ifp->state == INET6_IFADDR_STATE_DEAD && -		    ip6_del_rt(ifp->rt)) +		if (ip6_del_rt(ifp->rt))  			dst_free(&ifp->rt->dst);  		break;  	} +	atomic_inc(&net->ipv6.dev_addr_genid); +	rt_genid_bump_ipv6(net);  }  static void ipv6_ifa_notify(int event, struct inet6_ifaddr *ifp) @@ -4176,15 +4768,23 @@ static void ipv6_ifa_notify(int event, struct inet6_ifaddr *ifp)  #ifdef CONFIG_SYSCTL  static -int addrconf_sysctl_forward(ctl_table *ctl, int write, +int addrconf_sysctl_forward(struct ctl_table *ctl, int write,  			   void __user *buffer, size_t *lenp, loff_t *ppos)  {  	int *valp = ctl->data;  	int val = *valp;  	loff_t pos = *ppos; +	struct ctl_table lctl;  	int ret; -	ret = proc_dointvec(ctl, write, buffer, lenp, ppos); +	/* +	 * ctl->data points to idev->cnf.forwarding, we should +	 * not modify it until we get the rtnl lock. +	 */ +	lctl = *ctl; +	lctl.data = &val; + +	ret = proc_dointvec(&lctl, write, buffer, lenp, ppos);  	if (write)  		ret = addrconf_fixup_forwarding(ctl, valp, val); @@ -4195,13 +4795,16 @@ int addrconf_sysctl_forward(ctl_table *ctl, int write,  static void dev_disable_change(struct inet6_dev *idev)  { +	struct netdev_notifier_info info; +  	if (!idev || !idev->dev)  		return; +	netdev_notifier_info_init(&info, idev->dev);  	if (idev->cnf.disable_ipv6) -		addrconf_notify(NULL, NETDEV_DOWN, idev->dev); +		addrconf_notify(NULL, NETDEV_DOWN, &info);  	else -		addrconf_notify(NULL, NETDEV_UP, idev->dev); +		addrconf_notify(NULL, NETDEV_UP, &info);  }  static void addrconf_disable_change(struct net *net, __s32 newf) @@ -4222,26 +4825,27 @@ static void addrconf_disable_change(struct net *net, __s32 newf)  	rcu_read_unlock();  } -static int addrconf_disable_ipv6(struct ctl_table *table, int *p, int old) +static int addrconf_disable_ipv6(struct ctl_table *table, int *p, int newf)  {  	struct net *net; +	int old; + +	if (!rtnl_trylock()) +		return restart_syscall();  	net = (struct net *)table->extra2; +	old = *p; +	*p = newf; -	if (p == &net->ipv6.devconf_dflt->disable_ipv6) +	if (p == &net->ipv6.devconf_dflt->disable_ipv6) { +		rtnl_unlock();  		return 0; - -	if (!rtnl_trylock()) { -		/* Restore the original values before restarting */ -		*p = old; -		return restart_syscall();  	}  	if (p == &net->ipv6.devconf_all->disable_ipv6) { -		__s32 newf = net->ipv6.devconf_all->disable_ipv6;  		net->ipv6.devconf_dflt->disable_ipv6 = newf;  		addrconf_disable_change(net, newf); -	} else if ((!*p) ^ (!old)) +	} else if ((!newf) ^ (!old))  		dev_disable_change((struct inet6_dev *)table->extra1);  	rtnl_unlock(); @@ -4249,15 +4853,23 @@ static int addrconf_disable_ipv6(struct ctl_table *table, int *p, int old)  }  static -int addrconf_sysctl_disable(ctl_table *ctl, int write, +int addrconf_sysctl_disable(struct ctl_table *ctl, int write,  			    void __user *buffer, size_t *lenp, loff_t *ppos)  {  	int *valp = ctl->data;  	int val = *valp;  	loff_t pos = *ppos; +	struct ctl_table lctl;  	int ret; -	ret = proc_dointvec(ctl, write, buffer, lenp, ppos); +	/* +	 * ctl->data points to idev->cnf.disable_ipv6, we should +	 * not modify it until we get the rtnl lock. +	 */ +	lctl = *ctl; +	lctl.data = &val; + +	ret = proc_dointvec(&lctl, write, buffer, lenp, ppos);  	if (write)  		ret = addrconf_disable_ipv6(ctl, valp, val); @@ -4266,11 +4878,50 @@ int addrconf_sysctl_disable(ctl_table *ctl, int write,  	return ret;  } +static +int addrconf_sysctl_proxy_ndp(struct ctl_table *ctl, int write, +			      void __user *buffer, size_t *lenp, loff_t *ppos) +{ +	int *valp = ctl->data; +	int ret; +	int old, new; + +	old = *valp; +	ret = proc_dointvec(ctl, write, buffer, lenp, ppos); +	new = *valp; + +	if (write && old != new) { +		struct net *net = ctl->extra2; + +		if (!rtnl_trylock()) +			return restart_syscall(); + +		if (valp == &net->ipv6.devconf_dflt->proxy_ndp) +			inet6_netconf_notify_devconf(net, NETCONFA_PROXY_NEIGH, +						     NETCONFA_IFINDEX_DEFAULT, +						     net->ipv6.devconf_dflt); +		else if (valp == &net->ipv6.devconf_all->proxy_ndp) +			inet6_netconf_notify_devconf(net, NETCONFA_PROXY_NEIGH, +						     NETCONFA_IFINDEX_ALL, +						     net->ipv6.devconf_all); +		else { +			struct inet6_dev *idev = ctl->extra1; + +			inet6_netconf_notify_devconf(net, NETCONFA_PROXY_NEIGH, +						     idev->dev->ifindex, +						     &idev->cnf); +		} +		rtnl_unlock(); +	} + +	return ret; +} + +  static struct addrconf_sysctl_table  {  	struct ctl_table_header *sysctl_header; -	ctl_table addrconf_vars[DEVCONF_MAX+1]; -	char *dev_name; +	struct ctl_table addrconf_vars[DEVCONF_MAX+1];  } addrconf_sysctl __read_mostly = {  	.sysctl_header = NULL,  	.addrconf_vars = { @@ -4351,7 +5002,22 @@ static struct addrconf_sysctl_table  			.mode		= 0644,  			.proc_handler	= proc_dointvec,  		}, -#ifdef CONFIG_IPV6_PRIVACY +		{ +			.procname	= "mldv1_unsolicited_report_interval", +			.data		= +				&ipv6_devconf.mldv1_unsolicited_report_interval, +			.maxlen		= sizeof(int), +			.mode		= 0644, +			.proc_handler	= proc_dointvec_ms_jiffies, +		}, +		{ +			.procname	= "mldv2_unsolicited_report_interval", +			.data		= +				&ipv6_devconf.mldv2_unsolicited_report_interval, +			.maxlen		= sizeof(int), +			.mode		= 0644, +			.proc_handler	= proc_dointvec_ms_jiffies, +		},  		{  			.procname	= "use_tempaddr",  			.data		= &ipv6_devconf.use_tempaddr, @@ -4387,7 +5053,6 @@ static struct addrconf_sysctl_table  			.mode		= 0644,  			.proc_handler	= proc_dointvec,  		}, -#endif  		{  			.procname	= "max_addresses",  			.data		= &ipv6_devconf.max_addresses, @@ -4439,7 +5104,7 @@ static struct addrconf_sysctl_table  			.data		= &ipv6_devconf.proxy_ndp,  			.maxlen		= sizeof(int),  			.mode		= 0644, -			.proc_handler	= proc_dointvec, +			.proc_handler	= addrconf_sysctl_proxy_ndp,  		},  		{  			.procname	= "accept_source_route", @@ -4489,6 +5154,20 @@ static struct addrconf_sysctl_table  			.proc_handler   = proc_dointvec  		},  		{ +			.procname       = "ndisc_notify", +			.data           = &ipv6_devconf.ndisc_notify, +			.maxlen         = sizeof(int), +			.mode           = 0644, +			.proc_handler   = proc_dointvec +		}, +		{ +			.procname	= "suppress_frag_ndisc", +			.data		= &ipv6_devconf.suppress_frag_ndisc, +			.maxlen		= sizeof(int), +			.mode		= 0644, +			.proc_handler	= proc_dointvec +		}, +		{  			/* sentinel */  		}  	}, @@ -4499,17 +5178,7 @@ static int __addrconf_sysctl_register(struct net *net, char *dev_name,  {  	int i;  	struct addrconf_sysctl_table *t; - -#define ADDRCONF_CTL_PATH_DEV	3 - -	struct ctl_path addrconf_ctl_path[] = { -		{ .procname = "net", }, -		{ .procname = "ipv6", }, -		{ .procname = "conf", }, -		{ /* to be set */ }, -		{ }, -	}; - +	char path[sizeof("net/ipv6/conf/") + IFNAMSIZ];  	t = kmemdup(&addrconf_sysctl, sizeof(*t), GFP_KERNEL);  	if (t == NULL) @@ -4521,27 +5190,15 @@ static int __addrconf_sysctl_register(struct net *net, char *dev_name,  		t->addrconf_vars[i].extra2 = net;  	} -	/* -	 * Make a copy of dev_name, because '.procname' is regarded as const -	 * by sysctl and we wouldn't want anyone to change it under our feet -	 * (see SIOCSIFNAME). -	 */ -	t->dev_name = kstrdup(dev_name, GFP_KERNEL); -	if (!t->dev_name) -		goto free; +	snprintf(path, sizeof(path), "net/ipv6/conf/%s", dev_name); -	addrconf_ctl_path[ADDRCONF_CTL_PATH_DEV].procname = t->dev_name; - -	t->sysctl_header = register_net_sysctl_table(net, addrconf_ctl_path, -			t->addrconf_vars); +	t->sysctl_header = register_net_sysctl(net, path, t->addrconf_vars);  	if (t->sysctl_header == NULL) -		goto free_procname; +		goto free;  	p->sysctl = t;  	return 0; -free_procname: -	kfree(t->dev_name);  free:  	kfree(t);  out: @@ -4557,14 +5214,13 @@ static void __addrconf_sysctl_unregister(struct ipv6_devconf *p)  	t = p->sysctl;  	p->sysctl = NULL; -	unregister_sysctl_table(t->sysctl_header); -	kfree(t->dev_name); +	unregister_net_sysctl_table(t->sysctl_header);  	kfree(t);  }  static void addrconf_sysctl_register(struct inet6_dev *idev)  { -	neigh_sysctl_register(idev->dev, idev->nd_parms, "ipv6", +	neigh_sysctl_register(idev->dev, idev->nd_parms,  			      &ndisc_ifinfo_sysctl_change);  	__addrconf_sysctl_register(dev_net(idev->dev), idev->dev->name,  					idev, &idev->cnf); @@ -4581,26 +5237,20 @@ static void addrconf_sysctl_unregister(struct inet6_dev *idev)  static int __net_init addrconf_init_net(struct net *net)  { -	int err; +	int err = -ENOMEM;  	struct ipv6_devconf *all, *dflt; -	err = -ENOMEM; -	all = &ipv6_devconf; -	dflt = &ipv6_devconf_dflt; +	all = kmemdup(&ipv6_devconf, sizeof(ipv6_devconf), GFP_KERNEL); +	if (all == NULL) +		goto err_alloc_all; -	if (!net_eq(net, &init_net)) { -		all = kmemdup(all, sizeof(ipv6_devconf), GFP_KERNEL); -		if (all == NULL) -			goto err_alloc_all; +	dflt = kmemdup(&ipv6_devconf_dflt, sizeof(ipv6_devconf_dflt), GFP_KERNEL); +	if (dflt == NULL) +		goto err_alloc_dflt; -		dflt = kmemdup(dflt, sizeof(ipv6_devconf_dflt), GFP_KERNEL); -		if (dflt == NULL) -			goto err_alloc_dflt; -	} else { -		/* these will be inherited by all namespaces */ -		dflt->autoconf = ipv6_defaults.autoconf; -		dflt->disable_ipv6 = ipv6_defaults.disable_ipv6; -	} +	/* these will be inherited by all namespaces */ +	dflt->autoconf = ipv6_defaults.autoconf; +	dflt->disable_ipv6 = ipv6_defaults.disable_ipv6;  	net->ipv6.devconf_all = all;  	net->ipv6.devconf_dflt = dflt; @@ -4645,26 +5295,11 @@ static struct pernet_operations addrconf_ops = {  	.exit = addrconf_exit_net,  }; -/* - *      Device notifier - */ - -int register_inet6addr_notifier(struct notifier_block *nb) -{ -	return atomic_notifier_chain_register(&inet6addr_chain, nb); -} -EXPORT_SYMBOL(register_inet6addr_notifier); - -int unregister_inet6addr_notifier(struct notifier_block *nb) -{ -	return atomic_notifier_chain_unregister(&inet6addr_chain, nb); -} -EXPORT_SYMBOL(unregister_inet6addr_notifier); -  static struct rtnl_af_ops inet6_ops = {  	.family		  = AF_INET6,  	.fill_link_af	  = inet6_fill_link_af,  	.get_link_af_size = inet6_get_link_af_size, +	.set_link_af	  = inet6_set_link_af,  };  /* @@ -4677,8 +5312,8 @@ int __init addrconf_init(void)  	err = ipv6_addr_label_init();  	if (err < 0) { -		printk(KERN_CRIT "IPv6 Addrconf:" -		       " cannot initialize default policy table: %d.\n", err); +		pr_crit("%s: cannot initialize default policy table: %d\n", +			__func__, err);  		goto out;  	} @@ -4686,6 +5321,12 @@ int __init addrconf_init(void)  	if (err < 0)  		goto out_addrlabel; +	addrconf_wq = create_workqueue("ipv6_addrconf"); +	if (!addrconf_wq) { +		err = -ENOMEM; +		goto out_nowq; +	} +  	/* The addrconf netdev notifier requires that loopback_dev  	 * has it's ipv6 private information allocated and setup  	 * before it can bring up and give link-local addresses @@ -4716,31 +5357,36 @@ int __init addrconf_init(void)  	register_netdevice_notifier(&ipv6_dev_notf); -	addrconf_verify(0); +	addrconf_verify(); -	err = rtnl_af_register(&inet6_ops); -	if (err < 0) -		goto errout_af; +	rtnl_af_register(&inet6_ops); -	err = __rtnl_register(PF_INET6, RTM_GETLINK, NULL, inet6_dump_ifinfo); +	err = __rtnl_register(PF_INET6, RTM_GETLINK, NULL, inet6_dump_ifinfo, +			      NULL);  	if (err < 0)  		goto errout;  	/* Only the first call to __rtnl_register can fail */ -	__rtnl_register(PF_INET6, RTM_NEWADDR, inet6_rtm_newaddr, NULL); -	__rtnl_register(PF_INET6, RTM_DELADDR, inet6_rtm_deladdr, NULL); -	__rtnl_register(PF_INET6, RTM_GETADDR, inet6_rtm_getaddr, inet6_dump_ifaddr); -	__rtnl_register(PF_INET6, RTM_GETMULTICAST, NULL, inet6_dump_ifmcaddr); -	__rtnl_register(PF_INET6, RTM_GETANYCAST, NULL, inet6_dump_ifacaddr); +	__rtnl_register(PF_INET6, RTM_NEWADDR, inet6_rtm_newaddr, NULL, NULL); +	__rtnl_register(PF_INET6, RTM_DELADDR, inet6_rtm_deladdr, NULL, NULL); +	__rtnl_register(PF_INET6, RTM_GETADDR, inet6_rtm_getaddr, +			inet6_dump_ifaddr, NULL); +	__rtnl_register(PF_INET6, RTM_GETMULTICAST, NULL, +			inet6_dump_ifmcaddr, NULL); +	__rtnl_register(PF_INET6, RTM_GETANYCAST, NULL, +			inet6_dump_ifacaddr, NULL); +	__rtnl_register(PF_INET6, RTM_GETNETCONF, inet6_netconf_get_devconf, +			inet6_netconf_dump_devconf, NULL);  	ipv6_addr_label_rtnl_register();  	return 0;  errout:  	rtnl_af_unregister(&inet6_ops); -errout_af:  	unregister_netdevice_notifier(&ipv6_dev_notf);  errlo: +	destroy_workqueue(addrconf_wq); +out_nowq:  	unregister_pernet_subsys(&addrconf_ops);  out_addrlabel:  	ipv6_addr_label_cleanup(); @@ -4776,7 +5422,8 @@ void addrconf_cleanup(void)  	for (i = 0; i < IN6_ADDR_HSIZE; i++)  		WARN_ON(!hlist_empty(&inet6_addr_lst[i]));  	spin_unlock_bh(&addrconf_hash_lock); - -	del_timer(&addr_chk_timer); +	cancel_delayed_work(&addr_chk_work);  	rtnl_unlock(); + +	destroy_workqueue(addrconf_wq);  }  | 
