diff options
Diffstat (limited to 'net/8021q/vlan.c')
-rw-r--r-- | net/8021q/vlan.c | 97 |
1 files changed, 34 insertions, 63 deletions
diff --git a/net/8021q/vlan.c b/net/8021q/vlan.c index a29c5ab5815c..97da977c2a23 100644 --- a/net/8021q/vlan.c +++ b/net/8021q/vlan.c | |||
@@ -22,6 +22,7 @@ | |||
22 | #include <linux/module.h> | 22 | #include <linux/module.h> |
23 | #include <linux/netdevice.h> | 23 | #include <linux/netdevice.h> |
24 | #include <linux/skbuff.h> | 24 | #include <linux/skbuff.h> |
25 | #include <linux/slab.h> | ||
25 | #include <linux/init.h> | 26 | #include <linux/init.h> |
26 | #include <linux/rculist.h> | 27 | #include <linux/rculist.h> |
27 | #include <net/p8022.h> | 28 | #include <net/p8022.h> |
@@ -41,7 +42,7 @@ | |||
41 | 42 | ||
42 | /* Global VLAN variables */ | 43 | /* Global VLAN variables */ |
43 | 44 | ||
44 | int vlan_net_id; | 45 | int vlan_net_id __read_mostly; |
45 | 46 | ||
46 | /* Our listing of VLAN group(s) */ | 47 | /* Our listing of VLAN group(s) */ |
47 | static struct hlist_head vlan_group_hash[VLAN_GRP_HASH_SIZE]; | 48 | static struct hlist_head vlan_group_hash[VLAN_GRP_HASH_SIZE]; |
@@ -140,7 +141,7 @@ static void vlan_rcu_free(struct rcu_head *rcu) | |||
140 | vlan_group_free(container_of(rcu, struct vlan_group, rcu)); | 141 | vlan_group_free(container_of(rcu, struct vlan_group, rcu)); |
141 | } | 142 | } |
142 | 143 | ||
143 | void unregister_vlan_dev(struct net_device *dev) | 144 | void unregister_vlan_dev(struct net_device *dev, struct list_head *head) |
144 | { | 145 | { |
145 | struct vlan_dev_info *vlan = vlan_dev_info(dev); | 146 | struct vlan_dev_info *vlan = vlan_dev_info(dev); |
146 | struct net_device *real_dev = vlan->real_dev; | 147 | struct net_device *real_dev = vlan->real_dev; |
@@ -159,12 +160,13 @@ void unregister_vlan_dev(struct net_device *dev) | |||
159 | if (real_dev->features & NETIF_F_HW_VLAN_FILTER) | 160 | if (real_dev->features & NETIF_F_HW_VLAN_FILTER) |
160 | ops->ndo_vlan_rx_kill_vid(real_dev, vlan_id); | 161 | ops->ndo_vlan_rx_kill_vid(real_dev, vlan_id); |
161 | 162 | ||
162 | vlan_group_set_device(grp, vlan_id, NULL); | ||
163 | grp->nr_vlans--; | 163 | grp->nr_vlans--; |
164 | 164 | ||
165 | synchronize_net(); | 165 | vlan_group_set_device(grp, vlan_id, NULL); |
166 | if (!grp->killall) | ||
167 | synchronize_net(); | ||
166 | 168 | ||
167 | unregister_netdevice(dev); | 169 | unregister_netdevice_queue(dev, head); |
168 | 170 | ||
169 | /* If the group is now empty, kill off the group. */ | 171 | /* If the group is now empty, kill off the group. */ |
170 | if (grp->nr_vlans == 0) { | 172 | if (grp->nr_vlans == 0) { |
@@ -183,27 +185,6 @@ void unregister_vlan_dev(struct net_device *dev) | |||
183 | dev_put(real_dev); | 185 | dev_put(real_dev); |
184 | } | 186 | } |
185 | 187 | ||
186 | static void vlan_transfer_operstate(const struct net_device *dev, | ||
187 | struct net_device *vlandev) | ||
188 | { | ||
189 | /* Have to respect userspace enforced dormant state | ||
190 | * of real device, also must allow supplicant running | ||
191 | * on VLAN device | ||
192 | */ | ||
193 | if (dev->operstate == IF_OPER_DORMANT) | ||
194 | netif_dormant_on(vlandev); | ||
195 | else | ||
196 | netif_dormant_off(vlandev); | ||
197 | |||
198 | if (netif_carrier_ok(dev)) { | ||
199 | if (!netif_carrier_ok(vlandev)) | ||
200 | netif_carrier_on(vlandev); | ||
201 | } else { | ||
202 | if (netif_carrier_ok(vlandev)) | ||
203 | netif_carrier_off(vlandev); | ||
204 | } | ||
205 | } | ||
206 | |||
207 | int vlan_check_real_dev(struct net_device *real_dev, u16 vlan_id) | 188 | int vlan_check_real_dev(struct net_device *real_dev, u16 vlan_id) |
208 | { | 189 | { |
209 | const char *name = real_dev->name; | 190 | const char *name = real_dev->name; |
@@ -261,7 +242,7 @@ int register_vlan_dev(struct net_device *dev) | |||
261 | /* Account for reference in struct vlan_dev_info */ | 242 | /* Account for reference in struct vlan_dev_info */ |
262 | dev_hold(real_dev); | 243 | dev_hold(real_dev); |
263 | 244 | ||
264 | vlan_transfer_operstate(real_dev, dev); | 245 | netif_stacked_transfer_operstate(real_dev, dev); |
265 | linkwatch_fire_event(dev); /* _MUST_ call rfc2863_policy() */ | 246 | linkwatch_fire_event(dev); /* _MUST_ call rfc2863_policy() */ |
266 | 247 | ||
267 | /* So, got the sucker initialized, now lets place | 248 | /* So, got the sucker initialized, now lets place |
@@ -398,6 +379,8 @@ static void vlan_transfer_features(struct net_device *dev, | |||
398 | #if defined(CONFIG_FCOE) || defined(CONFIG_FCOE_MODULE) | 379 | #if defined(CONFIG_FCOE) || defined(CONFIG_FCOE_MODULE) |
399 | vlandev->fcoe_ddp_xid = dev->fcoe_ddp_xid; | 380 | vlandev->fcoe_ddp_xid = dev->fcoe_ddp_xid; |
400 | #endif | 381 | #endif |
382 | vlandev->real_num_tx_queues = dev->real_num_tx_queues; | ||
383 | BUG_ON(vlandev->real_num_tx_queues > vlandev->num_tx_queues); | ||
401 | 384 | ||
402 | if (old_features != vlandev->features) | 385 | if (old_features != vlandev->features) |
403 | netdev_features_change(vlandev); | 386 | netdev_features_change(vlandev); |
@@ -430,6 +413,8 @@ static int vlan_device_event(struct notifier_block *unused, unsigned long event, | |||
430 | struct vlan_group *grp; | 413 | struct vlan_group *grp; |
431 | int i, flgs; | 414 | int i, flgs; |
432 | struct net_device *vlandev; | 415 | struct net_device *vlandev; |
416 | struct vlan_dev_info *vlan; | ||
417 | LIST_HEAD(list); | ||
433 | 418 | ||
434 | if (is_vlan_dev(dev)) | 419 | if (is_vlan_dev(dev)) |
435 | __vlan_device_event(dev, event); | 420 | __vlan_device_event(dev, event); |
@@ -450,7 +435,7 @@ static int vlan_device_event(struct notifier_block *unused, unsigned long event, | |||
450 | if (!vlandev) | 435 | if (!vlandev) |
451 | continue; | 436 | continue; |
452 | 437 | ||
453 | vlan_transfer_operstate(dev, vlandev); | 438 | netif_stacked_transfer_operstate(dev, vlandev); |
454 | } | 439 | } |
455 | break; | 440 | break; |
456 | 441 | ||
@@ -505,8 +490,10 @@ static int vlan_device_event(struct notifier_block *unused, unsigned long event, | |||
505 | if (!(flgs & IFF_UP)) | 490 | if (!(flgs & IFF_UP)) |
506 | continue; | 491 | continue; |
507 | 492 | ||
508 | dev_change_flags(vlandev, flgs & ~IFF_UP); | 493 | vlan = vlan_dev_info(vlandev); |
509 | vlan_transfer_operstate(dev, vlandev); | 494 | if (!(vlan->flags & VLAN_FLAG_LOOSE_BINDING)) |
495 | dev_change_flags(vlandev, flgs & ~IFF_UP); | ||
496 | netif_stacked_transfer_operstate(dev, vlandev); | ||
510 | } | 497 | } |
511 | break; | 498 | break; |
512 | 499 | ||
@@ -521,13 +508,17 @@ static int vlan_device_event(struct notifier_block *unused, unsigned long event, | |||
521 | if (flgs & IFF_UP) | 508 | if (flgs & IFF_UP) |
522 | continue; | 509 | continue; |
523 | 510 | ||
524 | dev_change_flags(vlandev, flgs | IFF_UP); | 511 | vlan = vlan_dev_info(vlandev); |
525 | vlan_transfer_operstate(dev, vlandev); | 512 | if (!(vlan->flags & VLAN_FLAG_LOOSE_BINDING)) |
513 | dev_change_flags(vlandev, flgs | IFF_UP); | ||
514 | netif_stacked_transfer_operstate(dev, vlandev); | ||
526 | } | 515 | } |
527 | break; | 516 | break; |
528 | 517 | ||
529 | case NETDEV_UNREGISTER: | 518 | case NETDEV_UNREGISTER: |
530 | /* Delete all VLANs for this dev. */ | 519 | /* Delete all VLANs for this dev. */ |
520 | grp->killall = 1; | ||
521 | |||
531 | for (i = 0; i < VLAN_GROUP_ARRAY_LEN; i++) { | 522 | for (i = 0; i < VLAN_GROUP_ARRAY_LEN; i++) { |
532 | vlandev = vlan_group_get_device(grp, i); | 523 | vlandev = vlan_group_get_device(grp, i); |
533 | if (!vlandev) | 524 | if (!vlandev) |
@@ -538,8 +529,9 @@ static int vlan_device_event(struct notifier_block *unused, unsigned long event, | |||
538 | if (grp->nr_vlans == 1) | 529 | if (grp->nr_vlans == 1) |
539 | i = VLAN_GROUP_ARRAY_LEN; | 530 | i = VLAN_GROUP_ARRAY_LEN; |
540 | 531 | ||
541 | unregister_vlan_dev(vlandev); | 532 | unregister_vlan_dev(vlandev, &list); |
542 | } | 533 | } |
534 | unregister_netdevice_many(&list); | ||
543 | break; | 535 | break; |
544 | } | 536 | } |
545 | 537 | ||
@@ -645,7 +637,7 @@ static int vlan_ioctl_handler(struct net *net, void __user *arg) | |||
645 | err = -EPERM; | 637 | err = -EPERM; |
646 | if (!capable(CAP_NET_ADMIN)) | 638 | if (!capable(CAP_NET_ADMIN)) |
647 | break; | 639 | break; |
648 | unregister_vlan_dev(dev); | 640 | unregister_vlan_dev(dev, NULL); |
649 | err = 0; | 641 | err = 0; |
650 | break; | 642 | break; |
651 | 643 | ||
@@ -674,49 +666,28 @@ out: | |||
674 | return err; | 666 | return err; |
675 | } | 667 | } |
676 | 668 | ||
677 | static int vlan_init_net(struct net *net) | 669 | static int __net_init vlan_init_net(struct net *net) |
678 | { | 670 | { |
671 | struct vlan_net *vn = net_generic(net, vlan_net_id); | ||
679 | int err; | 672 | int err; |
680 | struct vlan_net *vn; | ||
681 | |||
682 | err = -ENOMEM; | ||
683 | vn = kzalloc(sizeof(struct vlan_net), GFP_KERNEL); | ||
684 | if (vn == NULL) | ||
685 | goto err_alloc; | ||
686 | |||
687 | err = net_assign_generic(net, vlan_net_id, vn); | ||
688 | if (err < 0) | ||
689 | goto err_assign; | ||
690 | 673 | ||
691 | vn->name_type = VLAN_NAME_TYPE_RAW_PLUS_VID_NO_PAD; | 674 | vn->name_type = VLAN_NAME_TYPE_RAW_PLUS_VID_NO_PAD; |
692 | 675 | ||
693 | err = vlan_proc_init(net); | 676 | err = vlan_proc_init(net); |
694 | if (err < 0) | ||
695 | goto err_proc; | ||
696 | |||
697 | return 0; | ||
698 | 677 | ||
699 | err_proc: | ||
700 | /* nothing */ | ||
701 | err_assign: | ||
702 | kfree(vn); | ||
703 | err_alloc: | ||
704 | return err; | 678 | return err; |
705 | } | 679 | } |
706 | 680 | ||
707 | static void vlan_exit_net(struct net *net) | 681 | static void __net_exit vlan_exit_net(struct net *net) |
708 | { | 682 | { |
709 | struct vlan_net *vn; | ||
710 | |||
711 | vn = net_generic(net, vlan_net_id); | ||
712 | rtnl_kill_links(net, &vlan_link_ops); | ||
713 | vlan_proc_cleanup(net); | 683 | vlan_proc_cleanup(net); |
714 | kfree(vn); | ||
715 | } | 684 | } |
716 | 685 | ||
717 | static struct pernet_operations vlan_net_ops = { | 686 | static struct pernet_operations vlan_net_ops = { |
718 | .init = vlan_init_net, | 687 | .init = vlan_init_net, |
719 | .exit = vlan_exit_net, | 688 | .exit = vlan_exit_net, |
689 | .id = &vlan_net_id, | ||
690 | .size = sizeof(struct vlan_net), | ||
720 | }; | 691 | }; |
721 | 692 | ||
722 | static int __init vlan_proto_init(void) | 693 | static int __init vlan_proto_init(void) |
@@ -726,7 +697,7 @@ static int __init vlan_proto_init(void) | |||
726 | pr_info("%s v%s %s\n", vlan_fullname, vlan_version, vlan_copyright); | 697 | pr_info("%s v%s %s\n", vlan_fullname, vlan_version, vlan_copyright); |
727 | pr_info("All bugs added by %s\n", vlan_buggyright); | 698 | pr_info("All bugs added by %s\n", vlan_buggyright); |
728 | 699 | ||
729 | err = register_pernet_gen_device(&vlan_net_id, &vlan_net_ops); | 700 | err = register_pernet_subsys(&vlan_net_ops); |
730 | if (err < 0) | 701 | if (err < 0) |
731 | goto err0; | 702 | goto err0; |
732 | 703 | ||
@@ -751,7 +722,7 @@ err4: | |||
751 | err3: | 722 | err3: |
752 | unregister_netdevice_notifier(&vlan_notifier_block); | 723 | unregister_netdevice_notifier(&vlan_notifier_block); |
753 | err2: | 724 | err2: |
754 | unregister_pernet_gen_device(vlan_net_id, &vlan_net_ops); | 725 | unregister_pernet_subsys(&vlan_net_ops); |
755 | err0: | 726 | err0: |
756 | return err; | 727 | return err; |
757 | } | 728 | } |
@@ -771,7 +742,7 @@ static void __exit vlan_cleanup_module(void) | |||
771 | for (i = 0; i < VLAN_GRP_HASH_SIZE; i++) | 742 | for (i = 0; i < VLAN_GRP_HASH_SIZE; i++) |
772 | BUG_ON(!hlist_empty(&vlan_group_hash[i])); | 743 | BUG_ON(!hlist_empty(&vlan_group_hash[i])); |
773 | 744 | ||
774 | unregister_pernet_gen_device(vlan_net_id, &vlan_net_ops); | 745 | unregister_pernet_subsys(&vlan_net_ops); |
775 | rcu_barrier(); /* Wait for completion of call_rcu()'s */ | 746 | rcu_barrier(); /* Wait for completion of call_rcu()'s */ |
776 | 747 | ||
777 | vlan_gvrp_uninit(); | 748 | vlan_gvrp_uninit(); |