2005-04-16 15:20:36 -07:00
/* -*- linux-c -*-
* INET 802.1 Q VLAN
* Ethernet - type device handling .
*
* Authors : Ben Greear < greearb @ candelatech . com >
2008-01-21 00:27:00 -08:00
* Please send support related email to : netdev @ vger . kernel . org
2005-04-16 15:20:36 -07:00
* VLAN Home Page : http : //www.candelatech.com/~greear/vlan.html
2007-02-09 23:24:25 +09:00
*
2005-04-16 15:20:36 -07:00
* Fixes : Mar 22 2001 : Martin Bokaemper < mbokaemper @ unispherenetworks . com >
* - reset skb - > pkt_type on incoming packets when MAC was changed
* - see that changed MAC is saddr for outgoing packets
* Oct 20 , 2001 : Ard van Breeman :
* - Fix MC - list , finally .
* - Flush MC - list on VLAN destroy .
2007-02-09 23:24:25 +09:00
*
2005-04-16 15:20:36 -07:00
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version
* 2 of the License , or ( at your option ) any later version .
*/
# include <linux/module.h>
# include <linux/mm.h>
# include <linux/in.h>
# include <linux/init.h>
# include <asm/uaccess.h> /* for copy_from_user */
# include <linux/skbuff.h>
# include <linux/netdevice.h>
# include <linux/etherdevice.h>
# include <net/datalink.h>
# include <net/p8022.h>
# include <net/arp.h>
# include "vlan.h"
# include "vlanproc.h"
# include <linux/if_vlan.h>
# include <net/ip.h>
/*
* Rebuild the Ethernet MAC header . This is called after an ARP
* ( or in future other address resolution ) has completed on this
* sk_buff . We now let ARP fill in the other fields .
*
* This routine CANNOT use cached dst - > neigh !
* Really , it is used only when dst - > neigh is wrong .
*
* TODO : This needs a checkup , I ' m ignorant here . - - BLG
*/
2008-01-21 00:22:11 -08:00
static int vlan_dev_rebuild_header ( struct sk_buff * skb )
2005-04-16 15:20:36 -07:00
{
struct net_device * dev = skb - > dev ;
struct vlan_ethhdr * veth = ( struct vlan_ethhdr * ) ( skb - > data ) ;
switch ( veth - > h_vlan_encapsulated_proto ) {
# ifdef CONFIG_INET
case __constant_htons ( ETH_P_IP ) :
/* TODO: Confirm this will work with VLAN headers... */
return arp_find ( veth - > h_dest , skb ) ;
2007-02-09 23:24:25 +09:00
# endif
2005-04-16 15:20:36 -07:00
default :
2008-01-21 00:24:30 -08:00
pr_debug ( " %s: unable to resolve type %X addresses. \n " ,
dev - > name , ntohs ( veth - > h_vlan_encapsulated_proto ) ) ;
2007-02-09 23:24:25 +09:00
2005-04-16 15:20:36 -07:00
memcpy ( veth - > h_source , dev - > dev_addr , ETH_ALEN ) ;
break ;
2007-04-20 17:09:22 -07:00
}
2005-04-16 15:20:36 -07:00
return 0 ;
}
static inline struct sk_buff * vlan_check_reorder_header ( struct sk_buff * skb )
{
2008-01-21 00:26:07 -08:00
if ( vlan_dev_info ( skb - > dev ) - > flags & VLAN_FLAG_REORDER_HDR ) {
2005-04-16 15:20:36 -07:00
if ( skb_shared ( skb ) | | skb_cloned ( skb ) ) {
struct sk_buff * nskb = skb_copy ( skb , GFP_ATOMIC ) ;
kfree_skb ( skb ) ;
skb = nskb ;
}
if ( skb ) {
/* Lifted from Gleb's VLAN code... */
memmove ( skb - > data - ETH_HLEN ,
skb - > data - VLAN_ETH_HLEN , 12 ) ;
2007-04-10 21:21:55 -07:00
skb - > mac_header + = VLAN_HLEN ;
2005-04-16 15:20:36 -07:00
}
}
return skb ;
}
2008-01-21 00:28:03 -08:00
static inline void vlan_set_encap_proto ( struct sk_buff * skb ,
struct vlan_hdr * vhdr )
{
__be16 proto ;
unsigned char * rawp ;
/*
* Was a VLAN packet , grab the encapsulated protocol , which the layer
* three protocols care about .
*/
proto = vhdr - > h_vlan_encapsulated_proto ;
if ( ntohs ( proto ) > = 1536 ) {
skb - > protocol = proto ;
return ;
}
rawp = skb - > data ;
if ( * ( unsigned short * ) rawp = = 0xFFFF )
/*
* This is a magic hack to spot IPX packets . Older Novell
* breaks the protocol design and runs IPX over 802.3 without
* an 802.2 LLC layer . We look for FFFF which isn ' t a used
* 802.2 SSAP / DSAP . This won ' t work for fault tolerant netware
* but does for the rest .
*/
skb - > protocol = htons ( ETH_P_802_3 ) ;
else
/*
* Real 802.2 LLC
*/
skb - > protocol = htons ( ETH_P_802_2 ) ;
}
2005-04-16 15:20:36 -07:00
/*
2007-02-09 23:24:25 +09:00
* Determine the packet ' s protocol ID . The rule here is that we
2005-04-16 15:20:36 -07:00
* assume 802.3 if the type field is short enough to be a length .
* This is normal practice and works for any ' now in use ' protocol .
*
* Also , at this point we assume that we ARE dealing exclusively with
* VLAN packets , or packets that should be made into VLAN packets based
* on a default VLAN ID .
*
* NOTE : Should be similar to ethernet / eth . c .
*
* SANITY NOTE : This method is called when a packet is moving up the stack
* towards userland . To get here , it would have already passed
* through the ethernet / eth . c eth_type_trans ( ) method .
* SANITY NOTE 2 : We are referencing to the VLAN_HDR frields , which MAY be
* stored UNALIGNED in the memory . RISC systems don ' t like
* such cases very much . . .
2008-01-21 00:26:41 -08:00
* SANITY NOTE 2 a : According to Dave Miller & Alexey , it will always be
* aligned , so there doesn ' t need to be any of the unaligned
* stuff . It has been commented out now . . . - - Ben
2005-04-16 15:20:36 -07:00
*
*/
int vlan_skb_recv ( struct sk_buff * skb , struct net_device * dev ,
2008-01-21 00:26:41 -08:00
struct packet_type * ptype , struct net_device * orig_dev )
2005-04-16 15:20:36 -07:00
{
2007-08-24 23:36:29 -07:00
struct vlan_hdr * vhdr ;
2005-04-16 15:20:36 -07:00
unsigned short vid ;
struct net_device_stats * stats ;
unsigned short vlan_TCI ;
2008-01-21 00:27:18 -08:00
if ( dev - > nd_net ! = & init_net )
goto err_free ;
2007-09-17 11:53:39 -07:00
2008-01-21 00:26:41 -08:00
skb = skb_share_check ( skb , GFP_ATOMIC ) ;
if ( skb = = NULL )
2008-01-21 00:27:18 -08:00
goto err_free ;
2007-08-24 23:36:29 -07:00
2008-01-21 00:27:18 -08:00
if ( unlikely ( ! pskb_may_pull ( skb , VLAN_HLEN ) ) )
goto err_free ;
2007-08-24 23:36:29 -07:00
2008-01-21 00:27:18 -08:00
vhdr = ( struct vlan_hdr * ) skb - > data ;
2005-04-16 15:20:36 -07:00
vlan_TCI = ntohs ( vhdr - > h_vlan_TCI ) ;
vid = ( vlan_TCI & VLAN_VID_MASK ) ;
rcu_read_lock ( ) ;
skb - > dev = __find_vlan_dev ( dev , vid ) ;
if ( ! skb - > dev ) {
2008-01-21 00:26:41 -08:00
pr_debug ( " %s: ERROR: No net_device for VID: %u on dev: %s \n " ,
__FUNCTION__ , ( unsigned int ) vid , dev - > name ) ;
2008-01-21 00:27:18 -08:00
goto err_unlock ;
2005-04-16 15:20:36 -07:00
}
skb - > dev - > last_rx = jiffies ;
2008-01-21 00:19:31 -08:00
stats = & skb - > dev - > stats ;
2005-04-16 15:20:36 -07:00
stats - > rx_packets + + ;
stats - > rx_bytes + = skb - > len ;
2006-03-20 22:43:56 -08:00
skb_pull_rcsum ( skb , VLAN_HLEN ) ;
2005-12-14 16:23:16 -08:00
2008-01-21 00:26:41 -08:00
skb - > priority = vlan_get_ingress_priority ( skb - > dev ,
ntohs ( vhdr - > h_vlan_TCI ) ) ;
2005-04-16 15:20:36 -07:00
2008-01-21 00:24:30 -08:00
pr_debug ( " %s: priority: %u for TCI: %hu \n " ,
__FUNCTION__ , skb - > priority , ntohs ( vhdr - > h_vlan_TCI ) ) ;
2005-04-16 15:20:36 -07:00
switch ( skb - > pkt_type ) {
case PACKET_BROADCAST : /* Yeah, stats collect these together.. */
2008-01-21 00:26:41 -08:00
/* stats->broadcast ++; // no such counter :-( */
2005-04-16 15:20:36 -07:00
break ;
case PACKET_MULTICAST :
stats - > multicast + + ;
break ;
2007-02-09 23:24:25 +09:00
case PACKET_OTHERHOST :
2005-04-16 15:20:36 -07:00
/* Our lower layer thinks this is not local, let's make sure.
2008-01-21 00:26:41 -08:00
* This allows the VLAN to have a different MAC than the
* underlying device , and still route correctly .
2005-04-16 15:20:36 -07:00
*/
2008-01-21 00:26:41 -08:00
if ( ! compare_ether_addr ( eth_hdr ( skb ) - > h_dest ,
skb - > dev - > dev_addr ) )
2005-04-16 15:20:36 -07:00
skb - > pkt_type = PACKET_HOST ;
break ;
default :
break ;
2007-04-20 17:09:22 -07:00
}
2005-04-16 15:20:36 -07:00
2008-01-21 00:28:03 -08:00
vlan_set_encap_proto ( skb , vhdr ) ;
2005-04-16 15:20:36 -07:00
skb = vlan_check_reorder_header ( skb ) ;
2008-01-21 00:27:18 -08:00
if ( ! skb ) {
2005-04-16 15:20:36 -07:00
stats - > rx_errors + + ;
2008-01-21 00:27:18 -08:00
goto err_unlock ;
2005-04-16 15:20:36 -07:00
}
2008-01-21 00:27:18 -08:00
netif_rx ( skb ) ;
2005-04-16 15:20:36 -07:00
rcu_read_unlock ( ) ;
2008-01-21 00:27:18 -08:00
return NET_RX_SUCCESS ;
err_unlock :
rcu_read_unlock ( ) ;
err_free :
kfree_skb ( skb ) ;
return NET_RX_DROP ;
2005-04-16 15:20:36 -07:00
}
2008-01-21 00:26:41 -08:00
static inline unsigned short
vlan_dev_get_egress_qos_mask ( struct net_device * dev , struct sk_buff * skb )
2005-04-16 15:20:36 -07:00
{
2008-01-21 00:26:41 -08:00
struct vlan_priority_tci_mapping * mp ;
2005-04-16 15:20:36 -07:00
2008-01-21 00:26:41 -08:00
mp = vlan_dev_info ( dev ) - > egress_priority_map [ ( skb - > priority & 0xF ) ] ;
2005-04-16 15:20:36 -07:00
while ( mp ) {
if ( mp - > priority = = skb - > priority ) {
2008-01-21 00:26:41 -08:00
return mp - > vlan_qos ; /* This should already be shifted
* to mask correctly with the
* VLAN ' s TCI */
2005-04-16 15:20:36 -07:00
}
mp = mp - > next ;
}
return 0 ;
}
/*
2007-02-09 23:24:25 +09:00
* Create the VLAN header for an arbitrary protocol layer
2005-04-16 15:20:36 -07:00
*
* saddr = NULL means use device source address
* daddr = NULL means leave destination address ( eg unresolved arp )
*
* This is called when the SKB is moving down the stack towards the
* physical devices .
*/
2008-01-21 00:22:11 -08:00
static int vlan_dev_hard_header ( struct sk_buff * skb , struct net_device * dev ,
unsigned short type ,
const void * daddr , const void * saddr ,
unsigned int len )
2005-04-16 15:20:36 -07:00
{
struct vlan_hdr * vhdr ;
unsigned short veth_TCI = 0 ;
int rc = 0 ;
int build_vlan_header = 0 ;
2008-01-21 00:26:41 -08:00
struct net_device * vdev = dev ;
2005-04-16 15:20:36 -07:00
2008-01-21 00:24:30 -08:00
pr_debug ( " %s: skb: %p type: %hx len: %u vlan_id: %hx, daddr: %p \n " ,
2008-01-21 00:26:41 -08:00
__FUNCTION__ , skb , type , len , vlan_dev_info ( dev ) - > vlan_id ,
daddr ) ;
2005-04-16 15:20:36 -07:00
/* build vlan header only if re_order_header flag is NOT set. This
* fixes some programs that get confused when they see a VLAN device
* sending a frame that is VLAN encoded ( the consensus is that the VLAN
* device should look completely like an Ethernet device when the
2007-02-09 23:24:25 +09:00
* REORDER_HEADER flag is set ) The drawback to this is some extra
2005-04-16 15:20:36 -07:00
* header shuffling in the hard_start_xmit . Users can turn off this
* REORDER behaviour with the vconfig tool .
*/
2008-01-21 00:26:07 -08:00
if ( ! ( vlan_dev_info ( dev ) - > flags & VLAN_FLAG_REORDER_HDR ) )
2007-06-13 12:07:37 -07:00
build_vlan_header = 1 ;
2005-04-16 15:20:36 -07:00
if ( build_vlan_header ) {
vhdr = ( struct vlan_hdr * ) skb_push ( skb , VLAN_HLEN ) ;
/* build the four bytes that make this a VLAN header. */
2008-01-21 00:26:41 -08:00
/* Now, construct the second two bytes. This field looks
* something like :
2005-04-16 15:20:36 -07:00
* usr_priority : 3 bits ( high bits )
* CFI 1 bit
* VLAN ID 12 bits ( low bits )
*
*/
2008-01-21 00:26:07 -08:00
veth_TCI = vlan_dev_info ( dev ) - > vlan_id ;
2005-04-16 15:20:36 -07:00
veth_TCI | = vlan_dev_get_egress_qos_mask ( dev , skb ) ;
vhdr - > h_vlan_TCI = htons ( veth_TCI ) ;
/*
2008-01-21 00:26:41 -08:00
* Set the protocol type . For a packet of type ETH_P_802_3 we
* put the length in here instead . It is up to the 802.2
* layer to carry protocol information .
2005-04-16 15:20:36 -07:00
*/
2008-01-21 00:26:41 -08:00
if ( type ! = ETH_P_802_3 )
2005-04-16 15:20:36 -07:00
vhdr - > h_vlan_encapsulated_proto = htons ( type ) ;
2008-01-21 00:26:41 -08:00
else
2005-04-16 15:20:36 -07:00
vhdr - > h_vlan_encapsulated_proto = htons ( len ) ;
2007-04-13 16:12:47 -07:00
skb - > protocol = htons ( ETH_P_8021Q ) ;
2007-04-19 20:34:51 -07:00
skb_reset_network_header ( skb ) ;
2005-04-16 15:20:36 -07:00
}
/* Before delegating work to the lower layer, enter our MAC-address */
if ( saddr = = NULL )
saddr = dev - > dev_addr ;
2008-01-21 00:26:07 -08:00
dev = vlan_dev_info ( dev ) - > real_dev ;
2005-04-16 15:20:36 -07:00
2008-01-21 00:26:41 -08:00
/* MPLS can send us skbuffs w/out enough space. This check will grow
* the skb if it doesn ' t have enough headroom . Not a beautiful solution ,
* so I ' ll tick a counter so that users can know it ' s happening . . .
* If they care . . .
2005-04-16 15:20:36 -07:00
*/
2008-01-21 00:26:41 -08:00
/* NOTE: This may still break if the underlying device is not the final
* device ( and thus there are more headers to add . . . ) It should work for
2005-04-16 15:20:36 -07:00
* good - ole - ethernet though .
*/
if ( skb_headroom ( skb ) < dev - > hard_header_len ) {
struct sk_buff * sk_tmp = skb ;
skb = skb_realloc_headroom ( sk_tmp , dev - > hard_header_len ) ;
kfree_skb ( sk_tmp ) ;
if ( skb = = NULL ) {
2008-01-21 00:19:31 -08:00
struct net_device_stats * stats = & vdev - > stats ;
2005-04-16 15:20:36 -07:00
stats - > tx_dropped + + ;
return - ENOMEM ;
}
2008-01-21 00:26:07 -08:00
vlan_dev_info ( vdev ) - > cnt_inc_headroom_on_tx + + ;
2008-01-21 00:26:41 -08:00
pr_debug ( " %s: %s: had to grow skb \n " , __FUNCTION__ , vdev - > name ) ;
2005-04-16 15:20:36 -07:00
}
if ( build_vlan_header ) {
/* Now make the underlying real hard header */
2007-10-09 01:36:32 -07:00
rc = dev_hard_header ( skb , dev , ETH_P_8021Q , daddr , saddr ,
len + VLAN_HLEN ) ;
if ( rc > 0 )
2005-04-16 15:20:36 -07:00
rc + = VLAN_HLEN ;
2007-10-09 01:36:32 -07:00
else if ( rc < 0 )
2005-04-16 15:20:36 -07:00
rc - = VLAN_HLEN ;
2007-10-09 01:36:32 -07:00
} else
2008-01-21 00:26:41 -08:00
/* If here, then we'll just make a normal looking ethernet
* frame , but , the hard_start_xmit method will insert the tag
* ( it has to be able to do this for bridged and other skbs
* that don ' t come down the protocol stack in an orderly manner .
2005-04-16 15:20:36 -07:00
*/
2007-10-09 01:36:32 -07:00
rc = dev_hard_header ( skb , dev , type , daddr , saddr , len ) ;
2005-04-16 15:20:36 -07:00
return rc ;
}
2008-01-21 00:22:11 -08:00
static int vlan_dev_hard_start_xmit ( struct sk_buff * skb , struct net_device * dev )
2005-04-16 15:20:36 -07:00
{
2008-01-21 00:19:31 -08:00
struct net_device_stats * stats = & dev - > stats ;
2005-04-16 15:20:36 -07:00
struct vlan_ethhdr * veth = ( struct vlan_ethhdr * ) ( skb - > data ) ;
2008-02-23 20:09:11 -08:00
2005-04-16 15:20:36 -07:00
/* Handle non-VLAN frames if they are sent to us, for example by DHCP.
*
* NOTE : THIS ASSUMES DIX ETHERNET , SPECIFICALLY NOT SUPPORTING
* OTHER THINGS LIKE FDDI / TokenRing / 802.3 SNAPs . . .
*/
2007-11-29 22:16:41 +11:00
if ( veth - > h_vlan_proto ! = htons ( ETH_P_8021Q ) | |
2008-01-21 00:26:07 -08:00
vlan_dev_info ( dev ) - > flags & VLAN_FLAG_REORDER_HDR ) {
2005-04-16 15:20:36 -07:00
int orig_headroom = skb_headroom ( skb ) ;
unsigned short veth_TCI ;
/* This is not a VLAN frame...but we can fix that! */
2008-01-21 00:26:07 -08:00
vlan_dev_info ( dev ) - > cnt_encap_on_xmit + + ;
2005-04-16 15:20:36 -07:00
2008-01-21 00:24:30 -08:00
pr_debug ( " %s: proto to encap: 0x%hx \n " ,
__FUNCTION__ , htons ( veth - > h_vlan_proto ) ) ;
2005-04-16 15:20:36 -07:00
/* Construct the second two bytes. This field looks something
* like :
* usr_priority : 3 bits ( high bits )
* CFI 1 bit
* VLAN ID 12 bits ( low bits )
*/
2008-01-21 00:26:07 -08:00
veth_TCI = vlan_dev_info ( dev ) - > vlan_id ;
2005-04-16 15:20:36 -07:00
veth_TCI | = vlan_dev_get_egress_qos_mask ( dev , skb ) ;
skb = __vlan_put_tag ( skb , veth_TCI ) ;
if ( ! skb ) {
stats - > tx_dropped + + ;
return 0 ;
}
2008-01-21 00:26:41 -08:00
if ( orig_headroom < VLAN_HLEN )
2008-01-21 00:26:07 -08:00
vlan_dev_info ( dev ) - > cnt_inc_headroom_on_tx + + ;
2005-04-16 15:20:36 -07:00
}
2008-01-21 00:24:30 -08:00
pr_debug ( " %s: about to send skb: %p to dev: %s \n " ,
2005-04-16 15:20:36 -07:00
__FUNCTION__ , skb , skb - > dev - > name ) ;
2008-02-23 20:09:11 -08:00
pr_debug ( " " MAC_FMT " " MAC_FMT " %4hx %4hx %4hx \n " ,
veth - > h_dest [ 0 ] , veth - > h_dest [ 1 ] , veth - > h_dest [ 2 ] ,
veth - > h_dest [ 3 ] , veth - > h_dest [ 4 ] , veth - > h_dest [ 5 ] ,
veth - > h_source [ 0 ] , veth - > h_source [ 1 ] , veth - > h_source [ 2 ] ,
veth - > h_source [ 3 ] , veth - > h_source [ 4 ] , veth - > h_source [ 5 ] ,
2008-01-21 00:24:30 -08:00
veth - > h_vlan_proto , veth - > h_vlan_TCI ,
veth - > h_vlan_encapsulated_proto ) ;
2005-04-16 15:20:36 -07:00
stats - > tx_packets + + ; /* for statics only */
stats - > tx_bytes + = skb - > len ;
2008-01-21 00:26:07 -08:00
skb - > dev = vlan_dev_info ( dev ) - > real_dev ;
2005-04-16 15:20:36 -07:00
dev_queue_xmit ( skb ) ;
return 0 ;
}
2008-01-21 00:22:11 -08:00
static int vlan_dev_hwaccel_hard_start_xmit ( struct sk_buff * skb ,
struct net_device * dev )
2005-04-16 15:20:36 -07:00
{
2008-01-21 00:19:31 -08:00
struct net_device_stats * stats = & dev - > stats ;
2005-04-16 15:20:36 -07:00
unsigned short veth_TCI ;
/* Construct the second two bytes. This field looks something
* like :
* usr_priority : 3 bits ( high bits )
* CFI 1 bit
* VLAN ID 12 bits ( low bits )
*/
2008-01-21 00:26:07 -08:00
veth_TCI = vlan_dev_info ( dev ) - > vlan_id ;
2005-04-16 15:20:36 -07:00
veth_TCI | = vlan_dev_get_egress_qos_mask ( dev , skb ) ;
skb = __vlan_hwaccel_put_tag ( skb , veth_TCI ) ;
stats - > tx_packets + + ;
stats - > tx_bytes + = skb - > len ;
2008-01-21 00:26:07 -08:00
skb - > dev = vlan_dev_info ( dev ) - > real_dev ;
2005-04-16 15:20:36 -07:00
dev_queue_xmit ( skb ) ;
return 0 ;
}
2008-01-21 00:22:11 -08:00
static int vlan_dev_change_mtu ( struct net_device * dev , int new_mtu )
2005-04-16 15:20:36 -07:00
{
/* TODO: gotta make sure the underlying layer can handle it,
* maybe an IFF_VLAN_CAPABLE flag for devices ?
*/
2008-01-21 00:26:07 -08:00
if ( vlan_dev_info ( dev ) - > real_dev - > mtu < new_mtu )
2005-04-16 15:20:36 -07:00
return - ERANGE ;
dev - > mtu = new_mtu ;
return 0 ;
}
2007-06-13 12:05:22 -07:00
void vlan_dev_set_ingress_priority ( const struct net_device * dev ,
u32 skb_prio , short vlan_prio )
2005-04-16 15:20:36 -07:00
{
2008-01-21 00:26:07 -08:00
struct vlan_dev_info * vlan = vlan_dev_info ( dev ) ;
2007-06-13 12:07:22 -07:00
if ( vlan - > ingress_priority_map [ vlan_prio & 0x7 ] & & ! skb_prio )
vlan - > nr_ingress_mappings - - ;
else if ( ! vlan - > ingress_priority_map [ vlan_prio & 0x7 ] & & skb_prio )
vlan - > nr_ingress_mappings + + ;
vlan - > ingress_priority_map [ vlan_prio & 0x7 ] = skb_prio ;
2005-04-16 15:20:36 -07:00
}
2007-06-13 12:05:22 -07:00
int vlan_dev_set_egress_priority ( const struct net_device * dev ,
u32 skb_prio , short vlan_prio )
2005-04-16 15:20:36 -07:00
{
2008-01-21 00:26:07 -08:00
struct vlan_dev_info * vlan = vlan_dev_info ( dev ) ;
2005-04-16 15:20:36 -07:00
struct vlan_priority_tci_mapping * mp = NULL ;
struct vlan_priority_tci_mapping * np ;
2007-06-13 12:07:22 -07:00
u32 vlan_qos = ( vlan_prio < < 13 ) & 0xE000 ;
2007-02-09 23:24:25 +09:00
2007-06-13 12:05:22 -07:00
/* See if a priority mapping exists.. */
2007-06-13 12:07:22 -07:00
mp = vlan - > egress_priority_map [ skb_prio & 0xF ] ;
2007-06-13 12:05:22 -07:00
while ( mp ) {
if ( mp - > priority = = skb_prio ) {
2007-06-13 12:07:22 -07:00
if ( mp - > vlan_qos & & ! vlan_qos )
vlan - > nr_egress_mappings - - ;
else if ( ! mp - > vlan_qos & & vlan_qos )
vlan - > nr_egress_mappings + + ;
mp - > vlan_qos = vlan_qos ;
2007-06-13 12:05:22 -07:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2007-06-13 12:05:22 -07:00
mp = mp - > next ;
2005-04-16 15:20:36 -07:00
}
2007-06-13 12:05:22 -07:00
/* Create a new mapping then. */
2007-06-13 12:07:22 -07:00
mp = vlan - > egress_priority_map [ skb_prio & 0xF ] ;
2007-06-13 12:05:22 -07:00
np = kmalloc ( sizeof ( struct vlan_priority_tci_mapping ) , GFP_KERNEL ) ;
if ( ! np )
return - ENOBUFS ;
np - > next = mp ;
np - > priority = skb_prio ;
2007-06-13 12:07:22 -07:00
np - > vlan_qos = vlan_qos ;
vlan - > egress_priority_map [ skb_prio & 0xF ] = np ;
if ( vlan_qos )
vlan - > nr_egress_mappings + + ;
2007-06-13 12:05:22 -07:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2007-06-13 12:07:37 -07:00
/* Flags are defined in the vlan_flags enum in include/linux/if_vlan.h file. */
2007-06-13 12:05:22 -07:00
int vlan_dev_set_vlan_flag ( const struct net_device * dev ,
u32 flag , short flag_val )
2005-04-16 15:20:36 -07:00
{
2007-06-13 12:05:22 -07:00
/* verify flag is supported */
2007-06-13 12:07:37 -07:00
if ( flag = = VLAN_FLAG_REORDER_HDR ) {
2008-01-21 00:26:41 -08:00
if ( flag_val )
2008-01-21 00:26:07 -08:00
vlan_dev_info ( dev ) - > flags | = VLAN_FLAG_REORDER_HDR ;
2008-01-21 00:26:41 -08:00
else
2008-01-21 00:26:07 -08:00
vlan_dev_info ( dev ) - > flags & = ~ VLAN_FLAG_REORDER_HDR ;
2007-06-13 12:05:22 -07:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
return - EINVAL ;
}
2007-06-13 12:05:22 -07:00
void vlan_dev_get_realdev_name ( const struct net_device * dev , char * result )
2005-04-16 15:20:36 -07:00
{
2008-01-21 00:26:07 -08:00
strncpy ( result , vlan_dev_info ( dev ) - > real_dev - > name , 23 ) ;
2005-04-16 15:20:36 -07:00
}
2007-06-13 12:05:22 -07:00
void vlan_dev_get_vid ( const struct net_device * dev , unsigned short * result )
2005-04-16 15:20:36 -07:00
{
2008-01-21 00:26:07 -08:00
* result = vlan_dev_info ( dev ) - > vlan_id ;
2005-04-16 15:20:36 -07:00
}
2008-01-21 00:22:11 -08:00
static int vlan_dev_open ( struct net_device * dev )
2005-04-16 15:20:36 -07:00
{
2008-01-21 00:26:07 -08:00
struct vlan_dev_info * vlan = vlan_dev_info ( dev ) ;
2007-07-11 19:45:24 -07:00
struct net_device * real_dev = vlan - > real_dev ;
int err ;
if ( ! ( real_dev - > flags & IFF_UP ) )
2005-04-16 15:20:36 -07:00
return - ENETDOWN ;
2007-07-11 19:45:24 -07:00
if ( compare_ether_addr ( dev - > dev_addr , real_dev - > dev_addr ) ) {
err = dev_unicast_add ( real_dev , dev - > dev_addr , ETH_ALEN ) ;
if ( err < 0 )
return err ;
}
memcpy ( vlan - > real_dev_addr , real_dev - > dev_addr , ETH_ALEN ) ;
2007-07-14 18:52:56 -07:00
if ( dev - > flags & IFF_ALLMULTI )
dev_set_allmulti ( real_dev , 1 ) ;
if ( dev - > flags & IFF_PROMISC )
dev_set_promiscuity ( real_dev , 1 ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2008-01-21 00:22:11 -08:00
static int vlan_dev_stop ( struct net_device * dev )
2005-04-16 15:20:36 -07:00
{
2008-01-21 00:26:07 -08:00
struct net_device * real_dev = vlan_dev_info ( dev ) - > real_dev ;
2007-07-11 19:45:24 -07:00
2007-07-14 18:53:28 -07:00
dev_mc_unsync ( real_dev , dev ) ;
2008-01-31 16:53:23 -08:00
dev_unicast_unsync ( real_dev , dev ) ;
2007-07-14 18:52:56 -07:00
if ( dev - > flags & IFF_ALLMULTI )
dev_set_allmulti ( real_dev , - 1 ) ;
if ( dev - > flags & IFF_PROMISC )
dev_set_promiscuity ( real_dev , - 1 ) ;
2007-07-11 19:45:24 -07:00
if ( compare_ether_addr ( dev - > dev_addr , real_dev - > dev_addr ) )
dev_unicast_delete ( real_dev , dev - > dev_addr , dev - > addr_len ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2008-01-21 00:22:11 -08:00
static int vlan_dev_set_mac_address ( struct net_device * dev , void * p )
2007-11-10 21:52:35 -08:00
{
2008-01-21 00:26:07 -08:00
struct net_device * real_dev = vlan_dev_info ( dev ) - > real_dev ;
2007-11-10 21:52:35 -08:00
struct sockaddr * addr = p ;
int err ;
if ( ! is_valid_ether_addr ( addr - > sa_data ) )
return - EADDRNOTAVAIL ;
if ( ! ( dev - > flags & IFF_UP ) )
goto out ;
if ( compare_ether_addr ( addr - > sa_data , real_dev - > dev_addr ) ) {
err = dev_unicast_add ( real_dev , addr - > sa_data , ETH_ALEN ) ;
if ( err < 0 )
return err ;
}
if ( compare_ether_addr ( dev - > dev_addr , real_dev - > dev_addr ) )
dev_unicast_delete ( real_dev , dev - > dev_addr , ETH_ALEN ) ;
out :
memcpy ( dev - > dev_addr , addr - > sa_data , ETH_ALEN ) ;
return 0 ;
}
2008-01-21 00:22:11 -08:00
static int vlan_dev_ioctl ( struct net_device * dev , struct ifreq * ifr , int cmd )
2005-04-16 15:20:36 -07:00
{
2008-01-21 00:26:07 -08:00
struct net_device * real_dev = vlan_dev_info ( dev ) - > real_dev ;
2005-04-16 15:20:36 -07:00
struct ifreq ifrr ;
int err = - EOPNOTSUPP ;
strncpy ( ifrr . ifr_name , real_dev - > name , IFNAMSIZ ) ;
ifrr . ifr_ifru = ifr - > ifr_ifru ;
2008-01-21 00:26:41 -08:00
switch ( cmd ) {
2005-04-16 15:20:36 -07:00
case SIOCGMIIPHY :
case SIOCGMIIREG :
case SIOCSMIIREG :
2007-02-09 23:24:25 +09:00
if ( real_dev - > do_ioctl & & netif_device_present ( real_dev ) )
2005-04-16 15:20:36 -07:00
err = real_dev - > do_ioctl ( real_dev , & ifrr , cmd ) ;
break ;
}
2007-02-09 23:24:25 +09:00
if ( ! err )
2005-04-16 15:20:36 -07:00
ifr - > ifr_ifru = ifrr . ifr_ifru ;
return err ;
}
2008-01-21 00:22:11 -08:00
static void vlan_dev_change_rx_flags ( struct net_device * dev , int change )
2007-07-14 18:52:56 -07:00
{
2008-01-21 00:26:07 -08:00
struct net_device * real_dev = vlan_dev_info ( dev ) - > real_dev ;
2007-07-14 18:52:56 -07:00
if ( change & IFF_ALLMULTI )
dev_set_allmulti ( real_dev , dev - > flags & IFF_ALLMULTI ? 1 : - 1 ) ;
if ( change & IFF_PROMISC )
dev_set_promiscuity ( real_dev , dev - > flags & IFF_PROMISC ? 1 : - 1 ) ;
}
2008-01-31 16:53:23 -08:00
static void vlan_dev_set_rx_mode ( struct net_device * vlan_dev )
2005-04-16 15:20:36 -07:00
{
2008-01-21 00:26:07 -08:00
dev_mc_sync ( vlan_dev_info ( vlan_dev ) - > real_dev , vlan_dev ) ;
2008-01-31 16:53:23 -08:00
dev_unicast_sync ( vlan_dev_info ( vlan_dev ) - > real_dev , vlan_dev ) ;
2005-04-16 15:20:36 -07:00
}
2008-01-21 00:22:11 -08:00
/*
* vlan network devices have devices nesting below it , and are a special
* " super class " of normal network devices ; split their locks off into a
* separate class since they always nest .
*/
static struct lock_class_key vlan_netdev_xmit_lock_key ;
static const struct header_ops vlan_header_ops = {
. create = vlan_dev_hard_header ,
. rebuild = vlan_dev_rebuild_header ,
. parse = eth_header_parse ,
} ;
static int vlan_dev_init ( struct net_device * dev )
{
2008-01-21 00:26:07 -08:00
struct net_device * real_dev = vlan_dev_info ( dev ) - > real_dev ;
2008-01-21 00:22:11 -08:00
int subclass = 0 ;
/* IFF_BROADCAST|IFF_MULTICAST; ??? */
dev - > flags = real_dev - > flags & ~ IFF_UP ;
dev - > iflink = real_dev - > ifindex ;
dev - > state = ( real_dev - > state & ( ( 1 < < __LINK_STATE_NOCARRIER ) |
( 1 < < __LINK_STATE_DORMANT ) ) ) |
( 1 < < __LINK_STATE_PRESENT ) ;
/* ipv6 shared card related stuff */
dev - > dev_id = real_dev - > dev_id ;
if ( is_zero_ether_addr ( dev - > dev_addr ) )
memcpy ( dev - > dev_addr , real_dev - > dev_addr , dev - > addr_len ) ;
if ( is_zero_ether_addr ( dev - > broadcast ) )
memcpy ( dev - > broadcast , real_dev - > broadcast , dev - > addr_len ) ;
if ( real_dev - > features & NETIF_F_HW_VLAN_TX ) {
dev - > header_ops = real_dev - > header_ops ;
dev - > hard_header_len = real_dev - > hard_header_len ;
dev - > hard_start_xmit = vlan_dev_hwaccel_hard_start_xmit ;
} else {
dev - > header_ops = & vlan_header_ops ;
dev - > hard_header_len = real_dev - > hard_header_len + VLAN_HLEN ;
dev - > hard_start_xmit = vlan_dev_hard_start_xmit ;
}
if ( real_dev - > priv_flags & IFF_802_1Q_VLAN )
subclass = 1 ;
lockdep_set_class_and_subclass ( & dev - > _xmit_lock ,
& vlan_netdev_xmit_lock_key , subclass ) ;
return 0 ;
}
void vlan_setup ( struct net_device * dev )
{
ether_setup ( dev ) ;
dev - > priv_flags | = IFF_802_1Q_VLAN ;
dev - > tx_queue_len = 0 ;
dev - > change_mtu = vlan_dev_change_mtu ;
dev - > init = vlan_dev_init ;
dev - > open = vlan_dev_open ;
dev - > stop = vlan_dev_stop ;
dev - > set_mac_address = vlan_dev_set_mac_address ;
2008-01-31 16:53:23 -08:00
dev - > set_rx_mode = vlan_dev_set_rx_mode ;
dev - > set_multicast_list = vlan_dev_set_rx_mode ;
2008-01-21 00:22:11 -08:00
dev - > change_rx_flags = vlan_dev_change_rx_flags ;
dev - > do_ioctl = vlan_dev_ioctl ;
dev - > destructor = free_netdev ;
memset ( dev - > broadcast , 0 , ETH_ALEN ) ;
}