diff options
Diffstat (limited to 'net/ipv6/addrconf.c')
| -rw-r--r-- | net/ipv6/addrconf.c | 70 | 
1 files changed, 35 insertions, 35 deletions
| diff --git a/net/ipv6/addrconf.c b/net/ipv6/addrconf.c index 27aed1afcf81..8ec4b3089e20 100644 --- a/net/ipv6/addrconf.c +++ b/net/ipv6/addrconf.c @@ -3176,35 +3176,9 @@ static void addrconf_gre_config(struct net_device *dev)  }  #endif -#if IS_ENABLED(CONFIG_NET_L3_MASTER_DEV) -/* If the host route is cached on the addr struct make sure it is associated - * with the proper table. e.g., enslavement can change and if so the cached - * host route needs to move to the new table. - */ -static void l3mdev_check_host_rt(struct inet6_dev *idev, -				  struct inet6_ifaddr *ifp) -{ -	if (ifp->rt) { -		u32 tb_id = l3mdev_fib_table(idev->dev) ? : RT6_TABLE_LOCAL; - -		if (tb_id != ifp->rt->rt6i_table->tb6_id) { -			ip6_del_rt(ifp->rt); -			ifp->rt = NULL; -		} -	} -} -#else -static void l3mdev_check_host_rt(struct inet6_dev *idev, -				  struct inet6_ifaddr *ifp) -{ -} -#endif -  static int fixup_permanent_addr(struct inet6_dev *idev,  				struct inet6_ifaddr *ifp)  { -	l3mdev_check_host_rt(idev, ifp); -  	if (!ifp->rt) {  		struct rt6_info *rt; @@ -3255,6 +3229,7 @@ static int addrconf_notify(struct notifier_block *this, unsigned long event,  			   void *ptr)  {  	struct net_device *dev = netdev_notifier_info_to_dev(ptr); +	struct netdev_notifier_changeupper_info *info;  	struct inet6_dev *idev = __in6_dev_get(dev);  	int run_pending = 0;  	int err; @@ -3303,6 +3278,9 @@ static int addrconf_notify(struct notifier_block *this, unsigned long event,  			break;  		if (event == NETDEV_UP) { +			/* restore routes for permanent addresses */ +			addrconf_permanent_addr(dev); +  			if (!addrconf_qdisc_ok(dev)) {  				/* device is not ready yet. */  				pr_info("ADDRCONF(NETDEV_UP): %s: link is not ready\n", @@ -3336,9 +3314,6 @@ static int addrconf_notify(struct notifier_block *this, unsigned long event,  			run_pending = 1;  		} -		/* restore routes for permanent addresses */ -		addrconf_permanent_addr(dev); -  		switch (dev->type) {  #if IS_ENABLED(CONFIG_IPV6_SIT)  		case ARPHRD_SIT: @@ -3413,6 +3388,15 @@ static int addrconf_notify(struct notifier_block *this, unsigned long event,  		if (idev)  			addrconf_type_change(dev, event);  		break; + +	case NETDEV_CHANGEUPPER: +		info = ptr; + +		/* flush all routes if dev is linked to or unlinked from +		 * an L3 master device (e.g., VRF) +		 */ +		if (info->upper_dev && netif_is_l3_master(info->upper_dev)) +			addrconf_ifdown(dev, 0);  	}  	return NOTIFY_OK; @@ -3438,6 +3422,12 @@ static void addrconf_type_change(struct net_device *dev, unsigned long event)  		ipv6_mc_unmap(idev);  } +static bool addr_is_local(const struct in6_addr *addr) +{ +	return ipv6_addr_type(addr) & +		(IPV6_ADDR_LINKLOCAL | IPV6_ADDR_LOOPBACK); +} +  static int addrconf_ifdown(struct net_device *dev, int how)  {  	struct net *net = dev_net(dev); @@ -3495,7 +3485,8 @@ restart:  				 * address is retained on a down event  				 */  				if (!keep_addr || -				    !(ifa->flags & IFA_F_PERMANENT)) { +				    !(ifa->flags & IFA_F_PERMANENT) || +				    addr_is_local(&ifa->addr)) {  					hlist_del_init_rcu(&ifa->addr_lst);  					goto restart;  				} @@ -3539,17 +3530,23 @@ restart:  	INIT_LIST_HEAD(&del_list);  	list_for_each_entry_safe(ifa, tmp, &idev->addr_list, if_list) { +		struct rt6_info *rt = NULL; +  		addrconf_del_dad_work(ifa);  		write_unlock_bh(&idev->lock);  		spin_lock_bh(&ifa->lock); -		if (keep_addr && (ifa->flags & IFA_F_PERMANENT)) { +		if (keep_addr && (ifa->flags & IFA_F_PERMANENT) && +		    !addr_is_local(&ifa->addr)) {  			/* set state to skip the notifier below */  			state = INET6_IFADDR_STATE_DEAD;  			ifa->state = 0;  			if (!(ifa->flags & IFA_F_NODAD))  				ifa->flags |= IFA_F_TENTATIVE; + +			rt = ifa->rt; +			ifa->rt = NULL;  		} else {  			state = ifa->state;  			ifa->state = INET6_IFADDR_STATE_DEAD; @@ -3560,6 +3557,9 @@ restart:  		spin_unlock_bh(&ifa->lock); +		if (rt) +			ip6_del_rt(rt); +  		if (state != INET6_IFADDR_STATE_DEAD) {  			__ipv6_ifa_notify(RTM_DELADDR, ifa);  			inet6addr_notifier_call_chain(NETDEV_DOWN, ifa); @@ -5325,10 +5325,10 @@ static void __ipv6_ifa_notify(int event, struct inet6_ifaddr *ifp)  			if (rt)  				ip6_del_rt(rt);  		} -		dst_hold(&ifp->rt->dst); - -		ip6_del_rt(ifp->rt); - +		if (ifp->rt) { +			dst_hold(&ifp->rt->dst); +			ip6_del_rt(ifp->rt); +		}  		rt_genid_bump_ipv6(net);  		break;  	} | 
