vlan_core.c 8.38 KB
Newer Older
1
2
3
#include <linux/skbuff.h>
#include <linux/netdevice.h>
#include <linux/if_vlan.h>
4
#include <linux/netpoll.h>
5
#include <linux/export.h>
6
7
#include "vlan.h"

8
bool vlan_do_receive(struct sk_buff **skbp)
9
{
10
11
	struct sk_buff *skb = *skbp;
	u16 vlan_id = skb->vlan_tci & VLAN_VID_MASK;
12
	struct net_device *vlan_dev;
Eric Dumazet's avatar
Eric Dumazet committed
13
	struct vlan_pcpu_stats *rx_stats;
14

15
	vlan_dev = vlan_find_dev(skb->dev, vlan_id);
16
	if (!vlan_dev)
17
		return false;
18

19
20
21
	skb = *skbp = skb_share_check(skb, GFP_ATOMIC);
	if (unlikely(!skb))
		return false;
Herbert Xu's avatar
Herbert Xu committed
22

23
	skb->dev = vlan_dev;
24
25
26
27
	if (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. */
28
		if (ether_addr_equal(eth_hdr(skb)->h_dest, vlan_dev->dev_addr))
29
30
31
			skb->pkt_type = PACKET_HOST;
	}

32
	if (!(vlan_dev_priv(vlan_dev)->flags & VLAN_FLAG_REORDER_HDR)) {
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
		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_tci);
		if (!skb)
			return false;
		skb_pull(skb, offset + VLAN_HLEN);
		skb_reset_mac_len(skb);
	}

48
	skb->priority = vlan_get_ingress_priority(vlan_dev, skb->vlan_tci);
49
	skb->vlan_tci = 0;
50

51
	rx_stats = this_cpu_ptr(vlan_dev_priv(vlan_dev)->vlan_pcpu_stats);
52

Eric Dumazet's avatar
Eric Dumazet committed
53
	u64_stats_update_begin(&rx_stats->syncp);
54
55
	rx_stats->rx_packets++;
	rx_stats->rx_bytes += skb->len;
56
	if (skb->pkt_type == PACKET_MULTICAST)
Eric Dumazet's avatar
Eric Dumazet committed
57
58
		rx_stats->rx_multicast++;
	u64_stats_update_end(&rx_stats->syncp);
59
60

	return true;
61
}
62

63
64
/* Must be invoked with rcu_read_lock. */
struct net_device *__vlan_find_dev_deep(struct net_device *dev,
65
66
					u16 vlan_id)
{
67
	struct vlan_info *vlan_info = rcu_dereference(dev->vlan_info);
68

69
70
	if (vlan_info) {
		return vlan_group_get_device(&vlan_info->grp, vlan_id);
71
72
	} else {
		/*
73
74
75
		 * Lower devices of master uppers (bonding, team) do not have
		 * grp assigned to themselves. Grp is assigned to upper device
		 * instead.
76
		 */
77
78
79
80
81
		struct net_device *upper_dev;

		upper_dev = netdev_master_upper_dev_get_rcu(dev);
		if (upper_dev)
			return __vlan_find_dev_deep(upper_dev, vlan_id);
82
83
84
85
86
87
	}

	return NULL;
}
EXPORT_SYMBOL(__vlan_find_dev_deep);

88
89
struct net_device *vlan_dev_real_dev(const struct net_device *dev)
{
90
	return vlan_dev_priv(dev)->real_dev;
91
}
92
EXPORT_SYMBOL(vlan_dev_real_dev);
93
94
95

u16 vlan_dev_vlan_id(const struct net_device *dev)
{
96
	return vlan_dev_priv(dev)->vlan_id;
97
}
98
EXPORT_SYMBOL(vlan_dev_vlan_id);
Herbert Xu's avatar
Herbert Xu committed
99

100
static struct sk_buff *vlan_reorder_header(struct sk_buff *skb)
101
{
102
103
104
105
	if (skb_cow(skb, skb_headroom(skb)) < 0)
		return NULL;
	memmove(skb->data - ETH_HLEN, skb->data - VLAN_ETH_HLEN, 2 * ETH_ALEN);
	skb->mac_header += VLAN_HLEN;
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
	return skb;
}

struct sk_buff *vlan_untag(struct sk_buff *skb)
{
	struct vlan_hdr *vhdr;
	u16 vlan_tci;

	if (unlikely(vlan_tx_tag_present(skb))) {
		/* vlan_tci is already set-up so leave this for another time */
		return skb;
	}

	skb = skb_share_check(skb, GFP_ATOMIC);
	if (unlikely(!skb))
		goto err_free;

	if (unlikely(!pskb_may_pull(skb, VLAN_HLEN)))
		goto err_free;

	vhdr = (struct vlan_hdr *) skb->data;
	vlan_tci = ntohs(vhdr->h_vlan_TCI);
	__vlan_hwaccel_put_tag(skb, vlan_tci);

	skb_pull_rcsum(skb, VLAN_HLEN);
	vlan_set_encap_proto(skb, vhdr);

133
	skb = vlan_reorder_header(skb);
134
135
136
	if (unlikely(!skb))
		goto err_free;

137
138
	skb_reset_network_header(skb);
	skb_reset_transport_header(skb);
139
140
	skb_reset_mac_len(skb);

141
142
143
144
145
146
	return skb;

err_free:
	kfree_skb(skb);
	return NULL;
}
147
EXPORT_SYMBOL(vlan_untag);
148

149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217

/*
 * vlan info and vid list
 */

static void vlan_group_free(struct vlan_group *grp)
{
	int i;

	for (i = 0; i < VLAN_GROUP_ARRAY_SPLIT_PARTS; i++)
		kfree(grp->vlan_devices_arrays[i]);
}

static void vlan_info_free(struct vlan_info *vlan_info)
{
	vlan_group_free(&vlan_info->grp);
	kfree(vlan_info);
}

static void vlan_info_rcu_free(struct rcu_head *rcu)
{
	vlan_info_free(container_of(rcu, struct vlan_info, rcu));
}

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;
}

struct vlan_vid_info {
	struct list_head list;
	unsigned short vid;
	int refcount;
};

static struct vlan_vid_info *vlan_vid_info_get(struct vlan_info *vlan_info,
					       unsigned short vid)
{
	struct vlan_vid_info *vid_info;

	list_for_each_entry(vid_info, &vlan_info->vid_list, list) {
		if (vid_info->vid == vid)
			return vid_info;
	}
	return NULL;
}

static struct vlan_vid_info *vlan_vid_info_alloc(unsigned short vid)
{
	struct vlan_vid_info *vid_info;

	vid_info = kzalloc(sizeof(struct vlan_vid_info), GFP_KERNEL);
	if (!vid_info)
		return NULL;
	vid_info->vid = vid;

	return vid_info;
}

static int __vlan_vid_add(struct vlan_info *vlan_info, unsigned short vid,
			  struct vlan_vid_info **pvid_info)
218
{
219
	struct net_device *dev = vlan_info->real_dev;
220
	const struct net_device_ops *ops = dev->netdev_ops;
221
222
223
224
225
226
	struct vlan_vid_info *vid_info;
	int err;

	vid_info = vlan_vid_info_alloc(vid);
	if (!vid_info)
		return -ENOMEM;
227

228
	if (dev->features & NETIF_F_HW_VLAN_FILTER) {
229
230
231
232
233
		err =  ops->ndo_vlan_rx_add_vid(dev, vid);
		if (err) {
			kfree(vid_info);
			return err;
		}
234
	}
235
236
237
	list_add(&vid_info->list, &vlan_info->vid_list);
	vlan_info->nr_vids++;
	*pvid_info = vid_info;
238
239
	return 0;
}
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274

int vlan_vid_add(struct net_device *dev, unsigned short 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, vid);
	if (!vid_info) {
		err = __vlan_vid_add(vlan_info, 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;
}
275
276
EXPORT_SYMBOL(vlan_vid_add);

277
278
static void __vlan_vid_del(struct vlan_info *vlan_info,
			   struct vlan_vid_info *vid_info)
279
{
280
	struct net_device *dev = vlan_info->real_dev;
281
	const struct net_device_ops *ops = dev->netdev_ops;
282
283
	unsigned short vid = vid_info->vid;
	int err;
284

285
	if (dev->features & NETIF_F_HW_VLAN_FILTER) {
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
		err = ops->ndo_vlan_rx_kill_vid(dev, vid);
		if (err) {
			pr_warn("failed to kill vid %d for device %s\n",
				vid, dev->name);
		}
	}
	list_del(&vid_info->list);
	kfree(vid_info);
	vlan_info->nr_vids--;
}

void vlan_vid_del(struct net_device *dev, unsigned short 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, 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);
		}
318
319
320
	}
}
EXPORT_SYMBOL(vlan_vid_del);
321
322
323
324
325

int vlan_vids_add_by_dev(struct net_device *dev,
			 const struct net_device *by_dev)
{
	struct vlan_vid_info *vid_info;
326
	struct vlan_info *vlan_info;
327
328
329
330
	int err;

	ASSERT_RTNL();

331
332
	vlan_info = rtnl_dereference(by_dev->vlan_info);
	if (!vlan_info)
333
334
		return 0;

335
	list_for_each_entry(vid_info, &vlan_info->vid_list, list) {
336
337
338
339
340
341
342
343
		err = vlan_vid_add(dev, vid_info->vid);
		if (err)
			goto unwind;
	}
	return 0;

unwind:
	list_for_each_entry_continue_reverse(vid_info,
344
					     &vlan_info->vid_list,
345
346
347
348
349
350
351
352
353
354
355
356
					     list) {
		vlan_vid_del(dev, vid_info->vid);
	}

	return err;
}
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;
357
	struct vlan_info *vlan_info;
358
359
360

	ASSERT_RTNL();

361
362
	vlan_info = rtnl_dereference(by_dev->vlan_info);
	if (!vlan_info)
363
364
		return;

365
	list_for_each_entry(vid_info, &vlan_info->vid_list, list)
366
367
368
		vlan_vid_del(dev, vid_info->vid);
}
EXPORT_SYMBOL(vlan_vids_del_by_dev);
369
370
371

bool vlan_uses_dev(const struct net_device *dev)
{
372
373
374
375
376
377
378
379
	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;
380
381
}
EXPORT_SYMBOL(vlan_uses_dev);