aboutsummaryrefslogtreecommitdiffstats
path: root/net/8021q/vlan_dev.c
diff options
context:
space:
mode:
Diffstat (limited to 'net/8021q/vlan_dev.c')
-rw-r--r--net/8021q/vlan_dev.c100
1 files changed, 86 insertions, 14 deletions
diff --git a/net/8021q/vlan_dev.c b/net/8021q/vlan_dev.c
index 9543e9182796..51ce4217c039 100644
--- a/net/8021q/vlan_dev.c
+++ b/net/8021q/vlan_dev.c
@@ -47,7 +47,7 @@
47 * 47 *
48 * TODO: This needs a checkup, I'm ignorant here. --BLG 48 * TODO: This needs a checkup, I'm ignorant here. --BLG
49 */ 49 */
50int vlan_dev_rebuild_header(struct sk_buff *skb) 50static int vlan_dev_rebuild_header(struct sk_buff *skb)
51{ 51{
52 struct net_device *dev = skb->dev; 52 struct net_device *dev = skb->dev;
53 struct vlan_ethhdr *veth = (struct vlan_ethhdr *)(skb->data); 53 struct vlan_ethhdr *veth = (struct vlan_ethhdr *)(skb->data);
@@ -342,9 +342,10 @@ static inline unsigned short vlan_dev_get_egress_qos_mask(struct net_device* dev
342 * This is called when the SKB is moving down the stack towards the 342 * This is called when the SKB is moving down the stack towards the
343 * physical devices. 343 * physical devices.
344 */ 344 */
345int vlan_dev_hard_header(struct sk_buff *skb, struct net_device *dev, 345static int vlan_dev_hard_header(struct sk_buff *skb, struct net_device *dev,
346 unsigned short type, 346 unsigned short type,
347 const void *daddr, const void *saddr, unsigned len) 347 const void *daddr, const void *saddr,
348 unsigned int len)
348{ 349{
349 struct vlan_hdr *vhdr; 350 struct vlan_hdr *vhdr;
350 unsigned short veth_TCI = 0; 351 unsigned short veth_TCI = 0;
@@ -451,7 +452,7 @@ int vlan_dev_hard_header(struct sk_buff *skb, struct net_device *dev,
451 return rc; 452 return rc;
452} 453}
453 454
454int vlan_dev_hard_start_xmit(struct sk_buff *skb, struct net_device *dev) 455static int vlan_dev_hard_start_xmit(struct sk_buff *skb, struct net_device *dev)
455{ 456{
456 struct net_device_stats *stats = &dev->stats; 457 struct net_device_stats *stats = &dev->stats;
457 struct vlan_ethhdr *veth = (struct vlan_ethhdr *)(skb->data); 458 struct vlan_ethhdr *veth = (struct vlan_ethhdr *)(skb->data);
@@ -512,7 +513,8 @@ int vlan_dev_hard_start_xmit(struct sk_buff *skb, struct net_device *dev)
512 return 0; 513 return 0;
513} 514}
514 515
515int vlan_dev_hwaccel_hard_start_xmit(struct sk_buff *skb, struct net_device *dev) 516static int vlan_dev_hwaccel_hard_start_xmit(struct sk_buff *skb,
517 struct net_device *dev)
516{ 518{
517 struct net_device_stats *stats = &dev->stats; 519 struct net_device_stats *stats = &dev->stats;
518 unsigned short veth_TCI; 520 unsigned short veth_TCI;
@@ -536,7 +538,7 @@ int vlan_dev_hwaccel_hard_start_xmit(struct sk_buff *skb, struct net_device *dev
536 return 0; 538 return 0;
537} 539}
538 540
539int vlan_dev_change_mtu(struct net_device *dev, int new_mtu) 541static int vlan_dev_change_mtu(struct net_device *dev, int new_mtu)
540{ 542{
541 /* TODO: gotta make sure the underlying layer can handle it, 543 /* TODO: gotta make sure the underlying layer can handle it,
542 * maybe an IFF_VLAN_CAPABLE flag for devices? 544 * maybe an IFF_VLAN_CAPABLE flag for devices?
@@ -626,7 +628,7 @@ void vlan_dev_get_vid(const struct net_device *dev, unsigned short *result)
626 *result = VLAN_DEV_INFO(dev)->vlan_id; 628 *result = VLAN_DEV_INFO(dev)->vlan_id;
627} 629}
628 630
629int vlan_dev_open(struct net_device *dev) 631static int vlan_dev_open(struct net_device *dev)
630{ 632{
631 struct vlan_dev_info *vlan = VLAN_DEV_INFO(dev); 633 struct vlan_dev_info *vlan = VLAN_DEV_INFO(dev);
632 struct net_device *real_dev = vlan->real_dev; 634 struct net_device *real_dev = vlan->real_dev;
@@ -650,7 +652,7 @@ int vlan_dev_open(struct net_device *dev)
650 return 0; 652 return 0;
651} 653}
652 654
653int vlan_dev_stop(struct net_device *dev) 655static int vlan_dev_stop(struct net_device *dev)
654{ 656{
655 struct net_device *real_dev = VLAN_DEV_INFO(dev)->real_dev; 657 struct net_device *real_dev = VLAN_DEV_INFO(dev)->real_dev;
656 658
@@ -666,7 +668,7 @@ int vlan_dev_stop(struct net_device *dev)
666 return 0; 668 return 0;
667} 669}
668 670
669int vlan_set_mac_address(struct net_device *dev, void *p) 671static int vlan_dev_set_mac_address(struct net_device *dev, void *p)
670{ 672{
671 struct net_device *real_dev = VLAN_DEV_INFO(dev)->real_dev; 673 struct net_device *real_dev = VLAN_DEV_INFO(dev)->real_dev;
672 struct sockaddr *addr = p; 674 struct sockaddr *addr = p;
@@ -692,7 +694,7 @@ out:
692 return 0; 694 return 0;
693} 695}
694 696
695int vlan_dev_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd) 697static int vlan_dev_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd)
696{ 698{
697 struct net_device *real_dev = VLAN_DEV_INFO(dev)->real_dev; 699 struct net_device *real_dev = VLAN_DEV_INFO(dev)->real_dev;
698 struct ifreq ifrr; 700 struct ifreq ifrr;
@@ -716,7 +718,7 @@ int vlan_dev_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd)
716 return err; 718 return err;
717} 719}
718 720
719void vlan_change_rx_flags(struct net_device *dev, int change) 721static void vlan_dev_change_rx_flags(struct net_device *dev, int change)
720{ 722{
721 struct net_device *real_dev = VLAN_DEV_INFO(dev)->real_dev; 723 struct net_device *real_dev = VLAN_DEV_INFO(dev)->real_dev;
722 724
@@ -726,8 +728,78 @@ void vlan_change_rx_flags(struct net_device *dev, int change)
726 dev_set_promiscuity(real_dev, dev->flags & IFF_PROMISC ? 1 : -1); 728 dev_set_promiscuity(real_dev, dev->flags & IFF_PROMISC ? 1 : -1);
727} 729}
728 730
729/** Taken from Gleb + Lennert's VLAN code, and modified... */ 731static void vlan_dev_set_multicast_list(struct net_device *vlan_dev)
730void vlan_dev_set_multicast_list(struct net_device *vlan_dev)
731{ 732{
732 dev_mc_sync(VLAN_DEV_INFO(vlan_dev)->real_dev, vlan_dev); 733 dev_mc_sync(VLAN_DEV_INFO(vlan_dev)->real_dev, vlan_dev);
733} 734}
735
736/*
737 * vlan network devices have devices nesting below it, and are a special
738 * "super class" of normal network devices; split their locks off into a
739 * separate class since they always nest.
740 */
741static struct lock_class_key vlan_netdev_xmit_lock_key;
742
743static const struct header_ops vlan_header_ops = {
744 .create = vlan_dev_hard_header,
745 .rebuild = vlan_dev_rebuild_header,
746 .parse = eth_header_parse,
747};
748
749static int vlan_dev_init(struct net_device *dev)
750{
751 struct net_device *real_dev = VLAN_DEV_INFO(dev)->real_dev;
752 int subclass = 0;
753
754 /* IFF_BROADCAST|IFF_MULTICAST; ??? */
755 dev->flags = real_dev->flags & ~IFF_UP;
756 dev->iflink = real_dev->ifindex;
757 dev->state = (real_dev->state & ((1<<__LINK_STATE_NOCARRIER) |
758 (1<<__LINK_STATE_DORMANT))) |
759 (1<<__LINK_STATE_PRESENT);
760
761 /* ipv6 shared card related stuff */
762 dev->dev_id = real_dev->dev_id;
763
764 if (is_zero_ether_addr(dev->dev_addr))
765 memcpy(dev->dev_addr, real_dev->dev_addr, dev->addr_len);
766 if (is_zero_ether_addr(dev->broadcast))
767 memcpy(dev->broadcast, real_dev->broadcast, dev->addr_len);
768
769 if (real_dev->features & NETIF_F_HW_VLAN_TX) {
770 dev->header_ops = real_dev->header_ops;
771 dev->hard_header_len = real_dev->hard_header_len;
772 dev->hard_start_xmit = vlan_dev_hwaccel_hard_start_xmit;
773 } else {
774 dev->header_ops = &vlan_header_ops;
775 dev->hard_header_len = real_dev->hard_header_len + VLAN_HLEN;
776 dev->hard_start_xmit = vlan_dev_hard_start_xmit;
777 }
778
779 if (real_dev->priv_flags & IFF_802_1Q_VLAN)
780 subclass = 1;
781
782 lockdep_set_class_and_subclass(&dev->_xmit_lock,
783 &vlan_netdev_xmit_lock_key, subclass);
784 return 0;
785}
786
787void vlan_setup(struct net_device *dev)
788{
789 ether_setup(dev);
790
791 dev->priv_flags |= IFF_802_1Q_VLAN;
792 dev->tx_queue_len = 0;
793
794 dev->change_mtu = vlan_dev_change_mtu;
795 dev->init = vlan_dev_init;
796 dev->open = vlan_dev_open;
797 dev->stop = vlan_dev_stop;
798 dev->set_mac_address = vlan_dev_set_mac_address;
799 dev->set_multicast_list = vlan_dev_set_multicast_list;
800 dev->change_rx_flags = vlan_dev_change_rx_flags;
801 dev->do_ioctl = vlan_dev_ioctl;
802 dev->destructor = free_netdev;
803
804 memset(dev->broadcast, 0, ETH_ALEN);
805}