Release 4.11 net/8021q/vlan_core.c
#include <linux/skbuff.h>
#include <linux/netdevice.h>
#include <linux/if_vlan.h>
#include <linux/netpoll.h>
#include <linux/export.h>
#include "vlan.h"
bool vlan_do_receive(struct sk_buff **skbp)
{
struct sk_buff *skb = *skbp;
__be16 vlan_proto = skb->vlan_proto;
u16 vlan_id = skb_vlan_tag_get_id(skb);
struct net_device *vlan_dev;
struct vlan_pcpu_stats *rx_stats;
vlan_dev = vlan_find_dev(skb->dev, vlan_proto, vlan_id);
if (!vlan_dev)
return false;
skb = *skbp = skb_share_check(skb, GFP_ATOMIC);
if (unlikely(!skb))
return false;
skb->dev = vlan_dev;
if (unlikely(skb->pkt_type == PACKET_OTHERHOST)) {
/* Our lower layer thinks this is not local, let's make sure.
* This allows the VLAN to have a different MAC than the
* underlying device, and still route correctly. */
if (ether_addr_equal_64bits(eth_hdr(skb)->h_dest, vlan_dev->dev_addr))
skb->pkt_type = PACKET_HOST;
}
if (!(vlan_dev_priv(vlan_dev)->flags & VLAN_FLAG_REORDER_HDR) &&
!netif_is_macvlan_port(vlan_dev) &&
!netif_is_bridge_port(vlan_dev)) {
unsigned int offset = skb->data - skb_mac_header(skb);
/*
* vlan_insert_tag expect skb->data pointing to mac header.
* So change skb->data before calling it and change back to
* original position later
*/
skb_push(skb, offset);
skb = *skbp = vlan_insert_tag(skb, skb->vlan_proto,
skb->vlan_tci);
if (!skb)
return false;
skb_pull(skb, offset + VLAN_HLEN);
skb_reset_mac_len(skb);
}
skb->priority = vlan_get_ingress_priority(vlan_dev, skb->vlan_tci);
skb->vlan_tci = 0;
rx_stats = this_cpu_ptr(vlan_dev_priv(vlan_dev)->vlan_pcpu_stats);
u64_stats_update_begin(&rx_stats->syncp);
rx_stats->rx_packets++;
rx_stats->rx_bytes += skb->len;
if (skb->pkt_type == PACKET_MULTICAST)
rx_stats->rx_multicast++;
u64_stats_update_end(&rx_stats->syncp);
return true;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jiri Pirko | 111 | 37.63% | 4 | 20.00% |
Patrick McHardy | 77 | 26.10% | 5 | 25.00% |
Jesse Gross | 45 | 15.25% | 1 | 5.00% |
Eric Dumazet | 30 | 10.17% | 4 | 20.00% |
Vlad Yasevich | 12 | 4.07% | 1 | 5.00% |
Pedro Garcia | 9 | 3.05% | 1 | 5.00% |
Herbert Xu | 6 | 2.03% | 1 | 5.00% |
Ding Tianhong | 4 | 1.36% | 2 | 10.00% |
Andy Gospodarek | 1 | 0.34% | 1 | 5.00% |
Total | 295 | 100.00% | 20 | 100.00% |
/* Must be invoked with rcu_read_lock. */
struct net_device *__vlan_find_dev_deep_rcu(struct net_device *dev,
__be16 vlan_proto, u16 vlan_id)
{
struct vlan_info *vlan_info = rcu_dereference(dev->vlan_info);
if (vlan_info) {
return vlan_group_get_device(&vlan_info->grp,
vlan_proto, vlan_id);
} else {
/*
* Lower devices of master uppers (bonding, team) do not have
* grp assigned to themselves. Grp is assigned to upper device
* instead.
*/
struct net_device *upper_dev;
upper_dev = netdev_master_upper_dev_get_rcu(dev);
if (upper_dev)
return __vlan_find_dev_deep_rcu(upper_dev,
vlan_proto, vlan_id);
}
return NULL;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jiri Pirko | 73 | 89.02% | 3 | 60.00% |
Patrick McHardy | 7 | 8.54% | 1 | 20.00% |
Ding Tianhong | 2 | 2.44% | 1 | 20.00% |
Total | 82 | 100.00% | 5 | 100.00% |
EXPORT_SYMBOL(__vlan_find_dev_deep_rcu);
struct net_device *vlan_dev_real_dev(const struct net_device *dev)
{
struct net_device *ret = vlan_dev_priv(dev)->real_dev;
while (is_vlan_dev(ret))
ret = vlan_dev_priv(ret)->real_dev;
return ret;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Nikolay Aleksandrov | 24 | 54.55% | 1 | 33.33% |
Patrick McHardy | 19 | 43.18% | 1 | 33.33% |
Jiri Pirko | 1 | 2.27% | 1 | 33.33% |
Total | 44 | 100.00% | 3 | 100.00% |
EXPORT_SYMBOL(vlan_dev_real_dev);
u16 vlan_dev_vlan_id(const struct net_device *dev)
{
return vlan_dev_priv(dev)->vlan_id;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Patrick McHardy | 18 | 94.74% | 1 | 50.00% |
Jiri Pirko | 1 | 5.26% | 1 | 50.00% |
Total | 19 | 100.00% | 2 | 100.00% |
EXPORT_SYMBOL(vlan_dev_vlan_id);
__be16 vlan_dev_vlan_proto(const struct net_device *dev)
{
return vlan_dev_priv(dev)->vlan_proto;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Ding Tianhong | 19 | 100.00% | 1 | 100.00% |
Total | 19 | 100.00% | 1 | 100.00% |
EXPORT_SYMBOL(vlan_dev_vlan_proto);
/*
* vlan info and vid list
*/
static void vlan_group_free(struct vlan_group *grp)
{
int i, j;
for (i = 0; i < VLAN_PROTO_NUM; i++)
for (j = 0; j < VLAN_GROUP_ARRAY_SPLIT_PARTS; j++)
kfree(grp->vlan_devices_arrays[i][j]);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jiri Pirko | 36 | 65.45% | 2 | 66.67% |
Patrick McHardy | 19 | 34.55% | 1 | 33.33% |
Total | 55 | 100.00% | 3 | 100.00% |
static void vlan_info_free(struct vlan_info *vlan_info)
{
vlan_group_free(&vlan_info->grp);
kfree(vlan_info);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jiri Pirko | 24 | 100.00% | 1 | 100.00% |
Total | 24 | 100.00% | 1 | 100.00% |
static void vlan_info_rcu_free(struct rcu_head *rcu)
{
vlan_info_free(container_of(rcu, struct vlan_info, rcu));
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jiri Pirko | 24 | 100.00% | 1 | 100.00% |
Total | 24 | 100.00% | 1 | 100.00% |
static struct vlan_info *vlan_info_alloc(struct net_device *dev)
{
struct vlan_info *vlan_info;
vlan_info = kzalloc(sizeof(struct vlan_info), GFP_KERNEL);
if (!vlan_info)
return NULL;
vlan_info->real_dev = dev;
INIT_LIST_HEAD(&vlan_info->vid_list);
return vlan_info;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jiri Pirko | 56 | 100.00% | 2 | 100.00% |
Total | 56 | 100.00% | 2 | 100.00% |
struct vlan_vid_info {
struct list_head list;
__be16 proto;
u16 vid;
int refcount;
};
static bool vlan_hw_filter_capable(const struct net_device *dev,
const struct vlan_vid_info *vid_info)
{
if (vid_info->proto == htons(ETH_P_8021Q) &&
dev->features & NETIF_F_HW_VLAN_CTAG_FILTER)
return true;
if (vid_info->proto == htons(ETH_P_8021AD) &&
dev->features & NETIF_F_HW_VLAN_STAG_FILTER)
return true;
return false;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Patrick McHardy | 61 | 100.00% | 1 | 100.00% |
Total | 61 | 100.00% | 1 | 100.00% |
static struct vlan_vid_info *vlan_vid_info_get(struct vlan_info *vlan_info,
__be16 proto, u16 vid)
{
struct vlan_vid_info *vid_info;
list_for_each_entry(vid_info, &vlan_info->vid_list, list) {
if (vid_info->proto == proto && vid_info->vid == vid)
return vid_info;
}
return NULL;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jiri Pirko | 44 | 81.48% | 2 | 66.67% |
Patrick McHardy | 10 | 18.52% | 1 | 33.33% |
Total | 54 | 100.00% | 3 | 100.00% |
static struct vlan_vid_info *vlan_vid_info_alloc(__be16 proto, u16 vid)
{
struct vlan_vid_info *vid_info;
vid_info = kzalloc(sizeof(struct vlan_vid_info), GFP_KERNEL);
if (!vid_info)
return NULL;
vid_info->proto = proto;
vid_info->vid = vid;
return vid_info;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jiri Pirko | 45 | 81.82% | 1 | 50.00% |
Patrick McHardy | 10 | 18.18% | 1 | 50.00% |
Total | 55 | 100.00% | 2 | 100.00% |
static int __vlan_vid_add(struct vlan_info *vlan_info, __be16 proto, u16 vid,
struct vlan_vid_info **pvid_info)
{
struct net_device *dev = vlan_info->real_dev;
const struct net_device_ops *ops = dev->netdev_ops;
struct vlan_vid_info *vid_info;
int err;
vid_info = vlan_vid_info_alloc(proto, vid);
if (!vid_info)
return -ENOMEM;
if (vlan_hw_filter_capable(dev, vid_info)) {
if (netif_device_present(dev))
err = ops->ndo_vlan_rx_add_vid(dev, proto, vid);
else
err = -ENODEV;
if (err) {
kfree(vid_info);
return err;
}
}
list_add(&vid_info->list, &vlan_info->vid_list);
vlan_info->nr_vids++;
*pvid_info = vid_info;
return 0;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jiri Pirko | 119 | 82.07% | 2 | 40.00% |
Padmanabh Ratnakar | 13 | 8.97% | 1 | 20.00% |
Patrick McHardy | 13 | 8.97% | 2 | 40.00% |
Total | 145 | 100.00% | 5 | 100.00% |
int vlan_vid_add(struct net_device *dev, __be16 proto, u16 vid)
{
struct vlan_info *vlan_info;
struct vlan_vid_info *vid_info;
bool vlan_info_created = false;
int err;
ASSERT_RTNL();
vlan_info = rtnl_dereference(dev->vlan_info);
if (!vlan_info) {
vlan_info = vlan_info_alloc(dev);
if (!vlan_info)
return -ENOMEM;
vlan_info_created = true;
}
vid_info = vlan_vid_info_get(vlan_info, proto, vid);
if (!vid_info) {
err = __vlan_vid_add(vlan_info, proto, vid, &vid_info);
if (err)
goto out_free_vlan_info;
}
vid_info->refcount++;
if (vlan_info_created)
rcu_assign_pointer(dev->vlan_info, vlan_info);
return 0;
out_free_vlan_info:
if (vlan_info_created)
kfree(vlan_info);
return err;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jiri Pirko | 139 | 94.56% | 1 | 50.00% |
Patrick McHardy | 8 | 5.44% | 1 | 50.00% |
Total | 147 | 100.00% | 2 | 100.00% |
EXPORT_SYMBOL(vlan_vid_add);
static void __vlan_vid_del(struct vlan_info *vlan_info,
struct vlan_vid_info *vid_info)
{
struct net_device *dev = vlan_info->real_dev;
const struct net_device_ops *ops = dev->netdev_ops;
__be16 proto = vid_info->proto;
u16 vid = vid_info->vid;
int err;
if (vlan_hw_filter_capable(dev, vid_info)) {
if (netif_device_present(dev))
err = ops->ndo_vlan_rx_kill_vid(dev, proto, vid);
else
err = -ENODEV;
if (err) {
pr_warn("failed to kill vid %04x/%d for device %s\n",
proto, vid, dev->name);
}
}
list_del(&vid_info->list);
kfree(vid_info);
vlan_info->nr_vids--;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jiri Pirko | 95 | 75.40% | 2 | 40.00% |
Patrick McHardy | 18 | 14.29% | 2 | 40.00% |
Padmanabh Ratnakar | 13 | 10.32% | 1 | 20.00% |
Total | 126 | 100.00% | 5 | 100.00% |
void vlan_vid_del(struct net_device *dev, __be16 proto, u16 vid)
{
struct vlan_info *vlan_info;
struct vlan_vid_info *vid_info;
ASSERT_RTNL();
vlan_info = rtnl_dereference(dev->vlan_info);
if (!vlan_info)
return;
vid_info = vlan_vid_info_get(vlan_info, proto, vid);
if (!vid_info)
return;
vid_info->refcount--;
if (vid_info->refcount == 0) {
__vlan_vid_del(vlan_info, vid_info);
if (vlan_info->nr_vids == 0) {
RCU_INIT_POINTER(dev->vlan_info, NULL);
call_rcu(&vlan_info->rcu, vlan_info_rcu_free);
}
}
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jiri Pirko | 106 | 94.64% | 2 | 66.67% |
Patrick McHardy | 6 | 5.36% | 1 | 33.33% |
Total | 112 | 100.00% | 3 | 100.00% |
EXPORT_SYMBOL(vlan_vid_del);
int vlan_vids_add_by_dev(struct net_device *dev,
const struct net_device *by_dev)
{
struct vlan_vid_info *vid_info;
struct vlan_info *vlan_info;
int err;
ASSERT_RTNL();
vlan_info = rtnl_dereference(by_dev->vlan_info);
if (!vlan_info)
return 0;
list_for_each_entry(vid_info, &vlan_info->vid_list, list) {
err = vlan_vid_add(dev, vid_info->proto, vid_info->vid);
if (err)
goto unwind;
}
return 0;
unwind:
list_for_each_entry_continue_reverse(vid_info,
&vlan_info->vid_list,
list) {
vlan_vid_del(dev, vid_info->proto, vid_info->vid);
}
return err;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jiri Pirko | 88 | 78.57% | 1 | 33.33% |
Dan Carpenter | 16 | 14.29% | 1 | 33.33% |
Patrick McHardy | 8 | 7.14% | 1 | 33.33% |
Total | 112 | 100.00% | 3 | 100.00% |
EXPORT_SYMBOL(vlan_vids_add_by_dev);
void vlan_vids_del_by_dev(struct net_device *dev,
const struct net_device *by_dev)
{
struct vlan_vid_info *vid_info;
struct vlan_info *vlan_info;
ASSERT_RTNL();
vlan_info = rtnl_dereference(by_dev->vlan_info);
if (!vlan_info)
return;
list_for_each_entry(vid_info, &vlan_info->vid_list, list)
vlan_vid_del(dev, vid_info->proto, vid_info->vid);
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jiri Pirko | 46 | 70.77% | 1 | 33.33% |
Dan Carpenter | 15 | 23.08% | 1 | 33.33% |
Patrick McHardy | 4 | 6.15% | 1 | 33.33% |
Total | 65 | 100.00% | 3 | 100.00% |
EXPORT_SYMBOL(vlan_vids_del_by_dev);
bool vlan_uses_dev(const struct net_device *dev)
{
struct vlan_info *vlan_info;
ASSERT_RTNL();
vlan_info = rtnl_dereference(dev->vlan_info);
if (!vlan_info)
return false;
return vlan_info->grp.nr_vlan_devs ? true : false;
}
Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jiri Pirko | 47 | 100.00% | 2 | 100.00% |
Total | 47 | 100.00% | 2 | 100.00% |
EXPORT_SYMBOL(vlan_uses_dev);
Overall Contributors
Person | Tokens | Prop | Commits | CommitProp |
Jiri Pirko | 1100 | 67.69% | 11 | 28.21% |
Patrick McHardy | 302 | 18.58% | 9 | 23.08% |
Jesse Gross | 45 | 2.77% | 1 | 2.56% |
Ding Tianhong | 31 | 1.91% | 4 | 10.26% |
Dan Carpenter | 31 | 1.91% | 1 | 2.56% |
Eric Dumazet | 30 | 1.85% | 4 | 10.26% |
Padmanabh Ratnakar | 26 | 1.60% | 1 | 2.56% |
Nikolay Aleksandrov | 24 | 1.48% | 1 | 2.56% |
Vlad Yasevich | 12 | 0.74% | 1 | 2.56% |
Pedro Garcia | 9 | 0.55% | 1 | 2.56% |
Herbert Xu | 9 | 0.55% | 2 | 5.13% |
Paul Gortmaker | 3 | 0.18% | 1 | 2.56% |
Ben Greear | 2 | 0.12% | 1 | 2.56% |
Andy Gospodarek | 1 | 0.06% | 1 | 2.56% |
Total | 1625 | 100.00% | 39 | 100.00% |
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.