diff options
Diffstat (limited to 'net/ipv6/addrconf.c')
-rw-r--r-- | net/ipv6/addrconf.c | 51 |
1 files changed, 28 insertions, 23 deletions
diff --git a/net/ipv6/addrconf.c b/net/ipv6/addrconf.c index 6dbf0f79b762..bcb55b78746f 100644 --- a/net/ipv6/addrconf.c +++ b/net/ipv6/addrconf.c | |||
@@ -2456,6 +2456,7 @@ static int addrconf_notify(struct notifier_block *this, unsigned long event, | |||
2456 | return notifier_from_errno(-ENOMEM); | 2456 | return notifier_from_errno(-ENOMEM); |
2457 | } | 2457 | } |
2458 | break; | 2458 | break; |
2459 | |||
2459 | case NETDEV_UP: | 2460 | case NETDEV_UP: |
2460 | case NETDEV_CHANGE: | 2461 | case NETDEV_CHANGE: |
2461 | if (dev->flags & IFF_SLAVE) | 2462 | if (dev->flags & IFF_SLAVE) |
@@ -2485,10 +2486,9 @@ static int addrconf_notify(struct notifier_block *this, unsigned long event, | |||
2485 | } | 2486 | } |
2486 | 2487 | ||
2487 | if (idev) { | 2488 | if (idev) { |
2488 | if (idev->if_flags & IF_READY) { | 2489 | if (idev->if_flags & IF_READY) |
2489 | /* device is already configured. */ | 2490 | /* device is already configured. */ |
2490 | break; | 2491 | break; |
2491 | } | ||
2492 | idev->if_flags |= IF_READY; | 2492 | idev->if_flags |= IF_READY; |
2493 | } | 2493 | } |
2494 | 2494 | ||
@@ -2517,25 +2517,30 @@ static int addrconf_notify(struct notifier_block *this, unsigned long event, | |||
2517 | addrconf_dev_config(dev); | 2517 | addrconf_dev_config(dev); |
2518 | break; | 2518 | break; |
2519 | } | 2519 | } |
2520 | |||
2520 | if (idev) { | 2521 | if (idev) { |
2521 | if (run_pending) | 2522 | if (run_pending) |
2522 | addrconf_dad_run(idev); | 2523 | addrconf_dad_run(idev); |
2523 | 2524 | ||
2524 | /* If the MTU changed during the interface down, when the | 2525 | /* |
2525 | interface up, the changed MTU must be reflected in the | 2526 | * If the MTU changed during the interface down, |
2526 | idev as well as routers. | 2527 | * when the interface up, the changed MTU must be |
2528 | * reflected in the idev as well as routers. | ||
2527 | */ | 2529 | */ |
2528 | if (idev->cnf.mtu6 != dev->mtu && dev->mtu >= IPV6_MIN_MTU) { | 2530 | if (idev->cnf.mtu6 != dev->mtu && |
2531 | dev->mtu >= IPV6_MIN_MTU) { | ||
2529 | rt6_mtu_change(dev, dev->mtu); | 2532 | rt6_mtu_change(dev, dev->mtu); |
2530 | idev->cnf.mtu6 = dev->mtu; | 2533 | idev->cnf.mtu6 = dev->mtu; |
2531 | } | 2534 | } |
2532 | idev->tstamp = jiffies; | 2535 | idev->tstamp = jiffies; |
2533 | inet6_ifinfo_notify(RTM_NEWLINK, idev); | 2536 | inet6_ifinfo_notify(RTM_NEWLINK, idev); |
2534 | /* If the changed mtu during down is lower than IPV6_MIN_MTU | 2537 | |
2535 | stop IPv6 on this interface. | 2538 | /* |
2539 | * If the changed mtu during down is lower than | ||
2540 | * IPV6_MIN_MTU stop IPv6 on this interface. | ||
2536 | */ | 2541 | */ |
2537 | if (dev->mtu < IPV6_MIN_MTU) | 2542 | if (dev->mtu < IPV6_MIN_MTU) |
2538 | addrconf_ifdown(dev, event != NETDEV_DOWN); | 2543 | addrconf_ifdown(dev, 1); |
2539 | } | 2544 | } |
2540 | break; | 2545 | break; |
2541 | 2546 | ||
@@ -2552,7 +2557,10 @@ static int addrconf_notify(struct notifier_block *this, unsigned long event, | |||
2552 | break; | 2557 | break; |
2553 | } | 2558 | } |
2554 | 2559 | ||
2555 | /* MTU falled under IPV6_MIN_MTU. Stop IPv6 on this interface. */ | 2560 | /* |
2561 | * MTU falled under IPV6_MIN_MTU. | ||
2562 | * Stop IPv6 on this interface. | ||
2563 | */ | ||
2556 | 2564 | ||
2557 | case NETDEV_DOWN: | 2565 | case NETDEV_DOWN: |
2558 | case NETDEV_UNREGISTER: | 2566 | case NETDEV_UNREGISTER: |
@@ -2572,6 +2580,7 @@ static int addrconf_notify(struct notifier_block *this, unsigned long event, | |||
2572 | return notifier_from_errno(err); | 2580 | return notifier_from_errno(err); |
2573 | } | 2581 | } |
2574 | break; | 2582 | break; |
2583 | |||
2575 | case NETDEV_PRE_TYPE_CHANGE: | 2584 | case NETDEV_PRE_TYPE_CHANGE: |
2576 | case NETDEV_POST_TYPE_CHANGE: | 2585 | case NETDEV_POST_TYPE_CHANGE: |
2577 | addrconf_type_change(dev, event); | 2586 | addrconf_type_change(dev, event); |
@@ -2586,7 +2595,6 @@ static int addrconf_notify(struct notifier_block *this, unsigned long event, | |||
2586 | */ | 2595 | */ |
2587 | static struct notifier_block ipv6_dev_notf = { | 2596 | static struct notifier_block ipv6_dev_notf = { |
2588 | .notifier_call = addrconf_notify, | 2597 | .notifier_call = addrconf_notify, |
2589 | .priority = 0 | ||
2590 | }; | 2598 | }; |
2591 | 2599 | ||
2592 | static void addrconf_type_change(struct net_device *dev, unsigned long event) | 2600 | static void addrconf_type_change(struct net_device *dev, unsigned long event) |
@@ -2618,8 +2626,9 @@ static int addrconf_ifdown(struct net_device *dev, int how) | |||
2618 | if (idev == NULL) | 2626 | if (idev == NULL) |
2619 | return -ENODEV; | 2627 | return -ENODEV; |
2620 | 2628 | ||
2621 | /* Step 1: remove reference to ipv6 device from parent device. | 2629 | /* |
2622 | Do not dev_put! | 2630 | * Step 1: remove reference to ipv6 device from parent device. |
2631 | * Do not dev_put! | ||
2623 | */ | 2632 | */ |
2624 | if (how) { | 2633 | if (how) { |
2625 | idev->dead = 1; | 2634 | idev->dead = 1; |
@@ -2634,16 +2643,15 @@ static int addrconf_ifdown(struct net_device *dev, int how) | |||
2634 | 2643 | ||
2635 | write_lock_bh(&idev->lock); | 2644 | write_lock_bh(&idev->lock); |
2636 | 2645 | ||
2637 | /* Step 3: clear flags for stateless addrconf */ | 2646 | /* Step 2: clear flags for stateless addrconf */ |
2638 | if (!how) | 2647 | if (!how) |
2639 | idev->if_flags &= ~(IF_RS_SENT|IF_RA_RCVD|IF_READY); | 2648 | idev->if_flags &= ~(IF_RS_SENT|IF_RA_RCVD|IF_READY); |
2640 | 2649 | ||
2641 | /* Step 4: clear address list */ | ||
2642 | #ifdef CONFIG_IPV6_PRIVACY | 2650 | #ifdef CONFIG_IPV6_PRIVACY |
2643 | if (how && del_timer(&idev->regen_timer)) | 2651 | if (how && del_timer(&idev->regen_timer)) |
2644 | in6_dev_put(idev); | 2652 | in6_dev_put(idev); |
2645 | 2653 | ||
2646 | /* clear tempaddr list */ | 2654 | /* Step 3: clear tempaddr list */ |
2647 | while (!list_empty(&idev->tempaddr_list)) { | 2655 | while (!list_empty(&idev->tempaddr_list)) { |
2648 | ifa = list_first_entry(&idev->tempaddr_list, | 2656 | ifa = list_first_entry(&idev->tempaddr_list, |
2649 | struct inet6_ifaddr, tmp_list); | 2657 | struct inet6_ifaddr, tmp_list); |
@@ -2669,7 +2677,7 @@ static int addrconf_ifdown(struct net_device *dev, int how) | |||
2669 | 2677 | ||
2670 | /* If just doing link down, and address is permanent | 2678 | /* If just doing link down, and address is permanent |
2671 | and not link-local, then retain it. */ | 2679 | and not link-local, then retain it. */ |
2672 | if (how == 0 && | 2680 | if (!how && |
2673 | (ifa->flags&IFA_F_PERMANENT) && | 2681 | (ifa->flags&IFA_F_PERMANENT) && |
2674 | !(ipv6_addr_type(&ifa->addr) & IPV6_ADDR_LINKLOCAL)) { | 2682 | !(ipv6_addr_type(&ifa->addr) & IPV6_ADDR_LINKLOCAL)) { |
2675 | list_move_tail(&ifa->if_list, &keep_list); | 2683 | list_move_tail(&ifa->if_list, &keep_list); |
@@ -2711,7 +2719,6 @@ static int addrconf_ifdown(struct net_device *dev, int how) | |||
2711 | write_unlock_bh(&idev->lock); | 2719 | write_unlock_bh(&idev->lock); |
2712 | 2720 | ||
2713 | /* Step 5: Discard multicast list */ | 2721 | /* Step 5: Discard multicast list */ |
2714 | |||
2715 | if (how) | 2722 | if (how) |
2716 | ipv6_mc_destroy_dev(idev); | 2723 | ipv6_mc_destroy_dev(idev); |
2717 | else | 2724 | else |
@@ -2719,8 +2726,7 @@ static int addrconf_ifdown(struct net_device *dev, int how) | |||
2719 | 2726 | ||
2720 | idev->tstamp = jiffies; | 2727 | idev->tstamp = jiffies; |
2721 | 2728 | ||
2722 | /* Shot the device (if unregistered) */ | 2729 | /* Last: Shot the device (if unregistered) */ |
2723 | |||
2724 | if (how) { | 2730 | if (how) { |
2725 | addrconf_sysctl_unregister(idev); | 2731 | addrconf_sysctl_unregister(idev); |
2726 | neigh_parms_release(&nd_tbl, idev->nd_parms); | 2732 | neigh_parms_release(&nd_tbl, idev->nd_parms); |
@@ -3108,8 +3114,7 @@ static void addrconf_verify(unsigned long foo) | |||
3108 | 3114 | ||
3109 | del_timer(&addr_chk_timer); | 3115 | del_timer(&addr_chk_timer); |
3110 | 3116 | ||
3111 | for (i=0; i < IN6_ADDR_HSIZE; i++) { | 3117 | for (i = 0; i < IN6_ADDR_HSIZE; i++) { |
3112 | |||
3113 | restart: | 3118 | restart: |
3114 | hlist_for_each_entry_rcu(ifp, node, | 3119 | hlist_for_each_entry_rcu(ifp, node, |
3115 | &inet6_addr_lst[i], addr_lst) { | 3120 | &inet6_addr_lst[i], addr_lst) { |
@@ -4376,7 +4381,7 @@ static int __addrconf_sysctl_register(struct net *net, char *dev_name, | |||
4376 | if (t == NULL) | 4381 | if (t == NULL) |
4377 | goto out; | 4382 | goto out; |
4378 | 4383 | ||
4379 | for (i=0; t->addrconf_vars[i].data; i++) { | 4384 | for (i = 0; t->addrconf_vars[i].data; i++) { |
4380 | t->addrconf_vars[i].data += (char*)p - (char*)&ipv6_devconf; | 4385 | t->addrconf_vars[i].data += (char*)p - (char*)&ipv6_devconf; |
4381 | t->addrconf_vars[i].extra1 = idev; /* embedded; no ref */ | 4386 | t->addrconf_vars[i].extra1 = idev; /* embedded; no ref */ |
4382 | t->addrconf_vars[i].extra2 = net; | 4387 | t->addrconf_vars[i].extra2 = net; |