Blame view

net/atm/br2684.c 21.7 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
3
4
5
6
7
  /*
  Experimental ethernet netdevice using ATM AAL5 as underlying carrier
  (RFC1483 obsoleted by RFC2684) for Linux 2.4
  Author: Marcell GAL, 2000, XDSL Ltd, Hungary
  */
  
  #include <linux/module.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
8
9
10
11
12
13
14
15
16
17
18
19
  #include <linux/init.h>
  #include <linux/kernel.h>
  #include <linux/list.h>
  #include <linux/netdevice.h>
  #include <linux/skbuff.h>
  #include <linux/etherdevice.h>
  #include <linux/rtnetlink.h>
  #include <linux/ip.h>
  #include <asm/uaccess.h>
  #include <net/arp.h>
  #include <linux/atm.h>
  #include <linux/atmdev.h>
4fc268d24   Randy Dunlap   [PATCH] capable/c...
20
  #include <linux/capability.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
21
22
23
24
25
  #include <linux/seq_file.h>
  
  #include <linux/atmbr2684.h>
  
  #include "common.h"
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
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
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
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
  
  /*
   * Define this to use a version of the code which interacts with the higher
   * layers in a more intellegent way, by always reserving enough space for
   * our header at the begining of the packet.  However, there may still be
   * some problems with programs like tcpdump.  In 2.5 we'll sort out what
   * we need to do to get this perfect.  For now we just will copy the packet
   * if we need space for the header
   */
  /* #define FASTER_VERSION */
  
  #ifdef DEBUG
  #define DPRINTK(format, args...) printk(KERN_DEBUG "br2684: " format, ##args)
  #else
  #define DPRINTK(format, args...)
  #endif
  
  #ifdef SKB_DEBUG
  static void skb_debug(const struct sk_buff *skb)
  {
  #define NUM2PRINT 50
  	char buf[NUM2PRINT * 3 + 1];	/* 3 chars per byte */
  	int i = 0;
  	for (i = 0; i < skb->len && i < NUM2PRINT; i++) {
  		sprintf(buf + i * 3, "%2.2x ", 0xff & skb->data[i]);
  	}
  	printk(KERN_DEBUG "br2684: skb: %s
  ", buf);
  }
  #else
  #define skb_debug(skb)	do {} while (0)
  #endif
  
  static unsigned char llc_oui_pid_pad[] =
      { 0xAA, 0xAA, 0x03, 0x00, 0x80, 0xC2, 0x00, 0x07, 0x00, 0x00 };
  #define PADLEN	(2)
  
  enum br2684_encaps {
  	e_vc  = BR2684_ENCAPS_VC,
  	e_llc = BR2684_ENCAPS_LLC,
  };
  
  struct br2684_vcc {
  	struct atm_vcc  *atmvcc;
  	struct net_device *device;
  	/* keep old push,pop functions for chaining */
  	void (*old_push)(struct atm_vcc *vcc,struct sk_buff *skb);
  	/* void (*old_pop)(struct atm_vcc *vcc,struct sk_buff *skb); */
  	enum br2684_encaps encaps;
  	struct list_head brvccs;
  #ifdef CONFIG_ATM_BR2684_IPFILTER
  	struct br2684_filter filter;
  #endif /* CONFIG_ATM_BR2684_IPFILTER */
  #ifndef FASTER_VERSION
  	unsigned copies_needed, copies_failed;
  #endif /* FASTER_VERSION */
  };
  
  struct br2684_dev {
  	struct net_device *net_dev;
  	struct list_head br2684_devs;
  	int number;
  	struct list_head brvccs; /* one device <=> one vcc (before xmas) */
  	struct net_device_stats stats;
  	int mac_was_set;
  };
  
  /*
   * This lock should be held for writing any time the list of devices or
   * their attached vcc's could be altered.  It should be held for reading
   * any time these are being queried.  Note that we sometimes need to
   * do read-locking under interrupt context, so write locking must block
   * the current CPU's interrupts
   */
  static DEFINE_RWLOCK(devs_lock);
  
  static LIST_HEAD(br2684_devs);
  
  static inline struct br2684_dev *BRPRIV(const struct net_device *net_dev)
  {
  	return (struct br2684_dev *) net_dev->priv;
  }
  
  static inline struct net_device *list_entry_brdev(const struct list_head *le)
  {
  	return list_entry(le, struct br2684_dev, br2684_devs)->net_dev;
  }
  
  static inline struct br2684_vcc *BR2684_VCC(const struct atm_vcc *atmvcc)
  {
  	return (struct br2684_vcc *) (atmvcc->user_back);
  }
  
  static inline struct br2684_vcc *list_entry_brvcc(const struct list_head *le)
  {
  	return list_entry(le, struct br2684_vcc, brvccs);
  }
  
  /* Caller should hold read_lock(&devs_lock) */
  static struct net_device *br2684_find_dev(const struct br2684_if_spec *s)
  {
  	struct list_head *lh;
  	struct net_device *net_dev;
  	switch (s->method) {
  	case BR2684_FIND_BYNUM:
  		list_for_each(lh, &br2684_devs) {
  			net_dev = list_entry_brdev(lh);
  			if (BRPRIV(net_dev)->number == s->spec.devnum)
  				return net_dev;
  		}
  		break;
  	case BR2684_FIND_BYIFNAME:
  		list_for_each(lh, &br2684_devs) {
  			net_dev = list_entry_brdev(lh);
  			if (!strncmp(net_dev->name, s->spec.ifname, IFNAMSIZ))
  				return net_dev;
  		}
  		break;
  	}
  	return NULL;
  }
  
  /*
   * Send a packet out a particular vcc.  Not to useful right now, but paves
   * the way for multiple vcc's per itf.  Returns true if we can send,
   * otherwise false
   */
  static int br2684_xmit_vcc(struct sk_buff *skb, struct br2684_dev *brdev,
  	struct br2684_vcc *brvcc)
  {
  	struct atm_vcc *atmvcc;
  #ifdef FASTER_VERSION
  	if (brvcc->encaps == e_llc)
  		memcpy(skb_push(skb, 8), llc_oui_pid_pad, 8);
  	/* last 2 bytes of llc_oui_pid_pad are managed by header routines;
  	   yes, you got it: 8 + 2 = sizeof(llc_oui_pid_pad)
  	 */
  #else
  	int minheadroom = (brvcc->encaps == e_llc) ? 10 : 2;
  	if (skb_headroom(skb) < minheadroom) {
  		struct sk_buff *skb2 = skb_realloc_headroom(skb, minheadroom);
  		brvcc->copies_needed++;
  		dev_kfree_skb(skb);
  		if (skb2 == NULL) {
  			brvcc->copies_failed++;
  			return 0;
  		}
  		skb = skb2;
  	}
  	skb_push(skb, minheadroom);
  	if (brvcc->encaps == e_llc)
27d7ff46a   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
177
  		skb_copy_to_linear_data(skb, llc_oui_pid_pad, 10);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
178
179
180
181
182
183
184
185
186
  	else
  		memset(skb->data, 0, 2);
  #endif /* FASTER_VERSION */
  	skb_debug(skb);
  
  	ATM_SKB(skb)->vcc = atmvcc = brvcc->atmvcc;
  	DPRINTK("atm_skb(%p)->vcc(%p)->dev(%p)
  ", skb, atmvcc, atmvcc->dev);
  	if (!atm_may_send(atmvcc, skb->truesize)) {
f7d57453d   YOSHIFUJI Hideaki   [NET] ATM: Fix wh...
187
  		/* we free this here for now, because we cannot know in a higher
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
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
218
219
220
221
222
223
224
225
  			layer whether the skb point it supplied wasn't freed yet.
  			now, it always is.
  		*/
  		dev_kfree_skb(skb);
  		return 0;
  		}
  	atomic_add(skb->truesize, &sk_atm(atmvcc)->sk_wmem_alloc);
  	ATM_SKB(skb)->atm_options = atmvcc->atm_options;
  	brdev->stats.tx_packets++;
  	brdev->stats.tx_bytes += skb->len;
  	atmvcc->send(atmvcc, skb);
  	return 1;
  }
  
  static inline struct br2684_vcc *pick_outgoing_vcc(struct sk_buff *skb,
  	struct br2684_dev *brdev)
  {
  	return list_empty(&brdev->brvccs) ? NULL :
  	    list_entry_brvcc(brdev->brvccs.next); /* 1 vcc/dev right now */
  }
  
  static int br2684_start_xmit(struct sk_buff *skb, struct net_device *dev)
  {
  	struct br2684_dev *brdev = BRPRIV(dev);
  	struct br2684_vcc *brvcc;
  
  	DPRINTK("br2684_start_xmit, skb->dst=%p
  ", skb->dst);
  	read_lock(&devs_lock);
  	brvcc = pick_outgoing_vcc(skb, brdev);
  	if (brvcc == NULL) {
  		DPRINTK("no vcc attached to dev %s
  ", dev->name);
  		brdev->stats.tx_errors++;
  		brdev->stats.tx_carrier_errors++;
  		/* netif_stop_queue(dev); */
  		dev_kfree_skb(skb);
  		read_unlock(&devs_lock);
4f55cd105   Jean-Denis Boyer   [ATM]: [br2684] i...
226
  		return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
227
228
229
230
231
232
233
234
235
236
237
238
239
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
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
  	}
  	if (!br2684_xmit_vcc(skb, brdev, brvcc)) {
  		/*
  		 * We should probably use netif_*_queue() here, but that
  		 * involves added complication.  We need to walk before
  		 * we can run
  		 */
  		/* don't free here! this pointer might be no longer valid!
  		dev_kfree_skb(skb);
  		*/
  		brdev->stats.tx_errors++;
  		brdev->stats.tx_fifo_errors++;
  	}
  	read_unlock(&devs_lock);
  	return 0;
  }
  
  static struct net_device_stats *br2684_get_stats(struct net_device *dev)
  {
  	DPRINTK("br2684_get_stats
  ");
  	return &BRPRIV(dev)->stats;
  }
  
  #ifdef FASTER_VERSION
  /*
   * These mirror eth_header and eth_header_cache.  They are not usually
   * exported for use in modules, so we grab them from net_device
   * after ether_setup() is done with it.  Bit of a hack.
   */
  static int (*my_eth_header)(struct sk_buff *, struct net_device *,
  	unsigned short, void *, void *, unsigned);
  static int (*my_eth_header_cache)(struct neighbour *, struct hh_cache *);
  
  static int
  br2684_header(struct sk_buff *skb, struct net_device *dev,
  	      unsigned short type, void *daddr, void *saddr, unsigned len)
  {
  	u16 *pad_before_eth;
  	int t = my_eth_header(skb, dev, type, daddr, saddr, len);
  	if (t > 0) {
  		pad_before_eth = (u16 *) skb_push(skb, 2);
  		*pad_before_eth = 0;
  		return dev->hard_header_len;	/* or return 16; ? */
  	} else
  		return t;
  }
  
  static int
  br2684_header_cache(struct neighbour *neigh, struct hh_cache *hh)
  {
  /* hh_data is 16 bytes long. if encaps is ether-llc we need 24, so
  xmit will add the additional header part in that case */
  	u16 *pad_before_eth = (u16 *)(hh->hh_data);
  	int t = my_eth_header_cache(neigh, hh);
  	DPRINTK("br2684_header_cache, neigh=%p, hh_cache=%p
  ", neigh, hh);
  	if (t < 0)
  		return t;
  	else {
  		*pad_before_eth = 0;
  		hh->hh_len = PADLEN + ETH_HLEN;
  	}
  	return 0;
  }
  
  /*
   * This is similar to eth_type_trans, which cannot be used because of
   * our dev->hard_header_len
   */
ab611487d   Alexey Dobriyan   [NET]: __be'ify *...
297
  static inline __be16 br_type_trans(struct sk_buff *skb, struct net_device *dev)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
298
299
300
301
  {
  	struct ethhdr *eth;
  	unsigned char *rawp;
  	eth = eth_hdr(skb);
dbbc09882   Kris Katterjohn   [NET]: Use newer ...
302
  	if (is_multicast_ether_addr(eth->h_dest)) {
d3f4a687f   Kris Katterjohn   [NET]: Change mem...
303
  		if (!compare_ether_addr(eth->h_dest, dev->broadcast))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
304
305
306
307
  			skb->pkt_type = PACKET_BROADCAST;
  		else
  			skb->pkt_type = PACKET_MULTICAST;
  	}
d3f4a687f   Kris Katterjohn   [NET]: Change mem...
308
  	else if (compare_ether_addr(eth->h_dest, dev->dev_addr))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
  		skb->pkt_type = PACKET_OTHERHOST;
  
  	if (ntohs(eth->h_proto) >= 1536)
  		return eth->h_proto;
  
  	rawp = skb->data;
  
  	/*
  	 * 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.
  	 */
  	if (*(unsigned short *) rawp == 0xFFFF)
  		return htons(ETH_P_802_3);
  
  	/*
  	 * Real 802.2 LLC
  	 */
  	return htons(ETH_P_802_2);
  }
  #endif /* FASTER_VERSION */
  
  /*
   * We remember when the MAC gets set, so we don't override it later with
   * the ESI of the ATM card of the first VC
   */
  static int (*my_eth_mac_addr)(struct net_device *, void *);
  static int br2684_mac_addr(struct net_device *dev, void *p)
  {
  	int err = my_eth_mac_addr(dev, p);
  	if (!err)
  		BRPRIV(dev)->mac_was_set = 1;
  	return err;
  }
  
  #ifdef CONFIG_ATM_BR2684_IPFILTER
  /* this IOCTL is experimental. */
  static int br2684_setfilt(struct atm_vcc *atmvcc, void __user *arg)
  {
  	struct br2684_vcc *brvcc;
  	struct br2684_filter_set fs;
  
  	if (copy_from_user(&fs, arg, sizeof fs))
  		return -EFAULT;
  	if (fs.ifspec.method != BR2684_FIND_BYNOTHING) {
  		/*
  		 * This is really a per-vcc thing, but we can also search
  		 * by device
  		 */
  		struct br2684_dev *brdev;
  		read_lock(&devs_lock);
  		brdev = BRPRIV(br2684_find_dev(&fs.ifspec));
  		if (brdev == NULL || list_empty(&brdev->brvccs) ||
  		    brdev->brvccs.next != brdev->brvccs.prev)  /* >1 VCC */
  			brvcc = NULL;
  		else
  			brvcc = list_entry_brvcc(brdev->brvccs.next);
  		read_unlock(&devs_lock);
  		if (brvcc == NULL)
  			return -ESRCH;
  	} else
  		brvcc = BR2684_VCC(atmvcc);
  	memcpy(&brvcc->filter, &fs.filter, sizeof(brvcc->filter));
  	return 0;
  }
  
  /* Returns 1 if packet should be dropped */
  static inline int
30d492da7   Al Viro   [ATM]: Annotations.
378
  packet_fails_filter(__be16 type, struct br2684_vcc *brvcc, struct sk_buff *skb)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
379
380
381
  {
  	if (brvcc->filter.netmask == 0)
  		return 0;			/* no filter in place */
acde4855b   YOSHIFUJI Hideaki   [NET] ATM: Use ht...
382
  	if (type == htons(ETH_P_IP) &&
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
383
384
385
  	    (((struct iphdr *) (skb->data))->daddr & brvcc->filter.
  	     netmask) == brvcc->filter.prefix)
  		return 0;
acde4855b   YOSHIFUJI Hideaki   [NET] ATM: Use ht...
386
  	if (type == htons(ETH_P_ARP))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
  		return 0;
  	/* TODO: we should probably filter ARPs too.. don't want to have
  	 *   them returning values that don't make sense, or is that ok?
  	 */
  	return 1;		/* drop */
  }
  #endif /* CONFIG_ATM_BR2684_IPFILTER */
  
  static void br2684_close_vcc(struct br2684_vcc *brvcc)
  {
  	DPRINTK("removing VCC %p from dev %p
  ", brvcc, brvcc->device);
  	write_lock_irq(&devs_lock);
  	list_del(&brvcc->brvccs);
  	write_unlock_irq(&devs_lock);
  	brvcc->atmvcc->user_back = NULL;	/* what about vcc->recvq ??? */
  	brvcc->old_push(brvcc->atmvcc, NULL);	/* pass on the bad news */
  	kfree(brvcc);
  	module_put(THIS_MODULE);
  }
  
  /* when AAL5 PDU comes in: */
  static void br2684_push(struct atm_vcc *atmvcc, struct sk_buff *skb)
  {
  	struct br2684_vcc *brvcc = BR2684_VCC(atmvcc);
  	struct net_device *net_dev = brvcc->device;
  	struct br2684_dev *brdev = BRPRIV(net_dev);
  	int plen = sizeof(llc_oui_pid_pad) + ETH_HLEN;
  
  	DPRINTK("br2684_push
  ");
  
  	if (unlikely(skb == NULL)) {
  		/* skb==NULL means VCC is being destroyed */
  		br2684_close_vcc(brvcc);
  		if (list_empty(&brdev->brvccs)) {
  			read_lock(&devs_lock);
  			list_del(&brdev->br2684_devs);
  			read_unlock(&devs_lock);
  			unregister_netdev(net_dev);
  			free_netdev(net_dev);
  		}
  		return;
  	}
  
  	skb_debug(skb);
  	atm_return(atmvcc, skb->truesize);
  	DPRINTK("skb from brdev %p
  ", brdev);
  	if (brvcc->encaps == e_llc) {
  		/* let us waste some time for checking the encapsulation.
  		   Note, that only 7 char is checked so frames with a valid FCS
  		   are also accepted (but FCS is not checked of course) */
  		if (memcmp(skb->data, llc_oui_pid_pad, 7)) {
  			brdev->stats.rx_errors++;
  			dev_kfree_skb(skb);
  			return;
  		}
  
  		/* Strip FCS if present */
  		if (skb->len > 7 && skb->data[7] == 0x01)
  			__skb_trim(skb, skb->len - 4);
  	} else {
  		plen = PADLEN + ETH_HLEN;	/* pad, dstmac,srcmac, ethtype */
  		/* first 2 chars should be 0 */
  		if (*((u16 *) (skb->data)) != 0) {
  			brdev->stats.rx_errors++;
  			dev_kfree_skb(skb);
  			return;
  		}
  	}
  	if (skb->len < plen) {
  		brdev->stats.rx_errors++;
  		dev_kfree_skb(skb);	/* dev_ not needed? */
  		return;
  	}
  
  #ifdef FASTER_VERSION
  	/* FIXME: tcpdump shows that pointer to mac header is 2 bytes earlier,
  	   than should be. What else should I set? */
  	skb_pull(skb, plen);
48d49d0cc   Arnaldo Carvalho de Melo   [SK_BUFF]: Introd...
468
  	skb_set_mac_header(skb, -ETH_HLEN);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
469
  	skb->pkt_type = PACKET_HOST;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
470
  	skb->protocol = br_type_trans(skb, net_dev);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
  #else
  	skb_pull(skb, plen - ETH_HLEN);
  	skb->protocol = eth_type_trans(skb, net_dev);
  #endif /* FASTER_VERSION */
  #ifdef CONFIG_ATM_BR2684_IPFILTER
  	if (unlikely(packet_fails_filter(skb->protocol, brvcc, skb))) {
  		brdev->stats.rx_dropped++;
  		dev_kfree_skb(skb);
  		return;
  	}
  #endif /* CONFIG_ATM_BR2684_IPFILTER */
  	skb->dev = net_dev;
  	ATM_SKB(skb)->vcc = atmvcc;	/* needed ? */
  	DPRINTK("received packet's protocol: %x
  ", ntohs(skb->protocol));
  	skb_debug(skb);
  	if (unlikely(!(net_dev->flags & IFF_UP))) {
  		/* sigh, interface is down */
  		brdev->stats.rx_dropped++;
  		dev_kfree_skb(skb);
  		return;
  	}
  	brdev->stats.rx_packets++;
  	brdev->stats.rx_bytes += skb->len;
  	memset(ATM_SKB(skb), 0, sizeof(struct atm_skb_data));
  	netif_rx(skb);
  }
  
  static int br2684_regvcc(struct atm_vcc *atmvcc, void __user *arg)
  {
  /* assign a vcc to a dev
  Note: we do not have explicit unassign, but look at _push()
  */
  	int err;
  	struct br2684_vcc *brvcc;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
506
  	struct sk_buff *skb;
c40a27f48   David S. Miller   [ATM]: Kill ipcom...
507
  	struct sk_buff_head *rq;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
508
509
510
  	struct br2684_dev *brdev;
  	struct net_device *net_dev;
  	struct atm_backend_br2684 be;
c40a27f48   David S. Miller   [ATM]: Kill ipcom...
511
  	unsigned long flags;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
512
513
514
  
  	if (copy_from_user(&be, arg, sizeof be))
  		return -EFAULT;
0da974f4f   Panagiotis Issaris   [NET]: Conversion...
515
  	brvcc = kzalloc(sizeof(struct br2684_vcc), GFP_KERNEL);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
516
517
  	if (!brvcc)
  		return -ENOMEM;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
  	write_lock_irq(&devs_lock);
  	net_dev = br2684_find_dev(&be.ifspec);
  	if (net_dev == NULL) {
  		printk(KERN_ERR
  		    "br2684: tried to attach to non-existant device
  ");
  		err = -ENXIO;
  		goto error;
  	}
  	brdev = BRPRIV(net_dev);
  	if (atmvcc->push == NULL) {
  		err = -EBADFD;
  		goto error;
  	}
  	if (!list_empty(&brdev->brvccs)) {
  		/* Only 1 VCC/dev right now */
  		err = -EEXIST;
  		goto error;
  	}
  	if (be.fcs_in != BR2684_FCSIN_NO || be.fcs_out != BR2684_FCSOUT_NO ||
  	    be.fcs_auto || be.has_vpiid || be.send_padding || (be.encaps !=
  	    BR2684_ENCAPS_VC && be.encaps != BR2684_ENCAPS_LLC) ||
  	    be.min_size != 0) {
  		err = -EINVAL;
  		goto error;
  	}
  	DPRINTK("br2684_regvcc vcc=%p, encaps=%d, brvcc=%p
  ", atmvcc, be.encaps,
  		brvcc);
  	if (list_empty(&brdev->brvccs) && !brdev->mac_was_set) {
  		unsigned char *esi = atmvcc->dev->esi;
  		if (esi[0] | esi[1] | esi[2] | esi[3] | esi[4] | esi[5])
  			memcpy(net_dev->dev_addr, esi, net_dev->addr_len);
  		else
  			net_dev->dev_addr[2] = 1;
  	}
  	list_add(&brvcc->brvccs, &brdev->brvccs);
  	write_unlock_irq(&devs_lock);
  	brvcc->device = net_dev;
  	brvcc->atmvcc = atmvcc;
  	atmvcc->user_back = brvcc;
  	brvcc->encaps = (enum br2684_encaps) be.encaps;
  	brvcc->old_push = atmvcc->push;
  	barrier();
  	atmvcc->push = br2684_push;
c40a27f48   David S. Miller   [ATM]: Kill ipcom...
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
  
  	rq = &sk_atm(atmvcc)->sk_receive_queue;
  
  	spin_lock_irqsave(&rq->lock, flags);
  	if (skb_queue_empty(rq)) {
  		skb = NULL;
  	} else {
  		/* NULL terminate the list.  */
  		rq->prev->next = NULL;
  		skb = rq->next;
  	}
  	rq->prev = rq->next = (struct sk_buff *)rq;
  	rq->qlen = 0;
  	spin_unlock_irqrestore(&rq->lock, flags);
  
  	while (skb) {
  		struct sk_buff *next = skb->next;
  
  		skb->next = skb->prev = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
582
583
584
  		BRPRIV(skb->dev)->stats.rx_bytes -= skb->len;
  		BRPRIV(skb->dev)->stats.rx_packets--;
  		br2684_push(atmvcc, skb);
c40a27f48   David S. Miller   [ATM]: Kill ipcom...
585
586
  
  		skb = next;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
  	}
  	__module_get(THIS_MODULE);
  	return 0;
      error:
  	write_unlock_irq(&devs_lock);
  	kfree(brvcc);
  	return err;
  }
  
  static void br2684_setup(struct net_device *netdev)
  {
  	struct br2684_dev *brdev = BRPRIV(netdev);
  
  	ether_setup(netdev);
  	brdev->net_dev = netdev;
  
  #ifdef FASTER_VERSION
  	my_eth_header = netdev->hard_header;
  	netdev->hard_header = br2684_header;
  	my_eth_header_cache = netdev->hard_header_cache;
  	netdev->hard_header_cache = br2684_header_cache;
  	netdev->hard_header_len = sizeof(llc_oui_pid_pad) + ETH_HLEN;	/* 10 + 14 */
  #endif
  	my_eth_mac_addr = netdev->set_mac_address;
  	netdev->set_mac_address = br2684_mac_addr;
  	netdev->hard_start_xmit = br2684_start_xmit;
  	netdev->get_stats = br2684_get_stats;
  
  	INIT_LIST_HEAD(&brdev->brvccs);
  }
  
  static int br2684_create(void __user *arg)
  {
  	int err;
  	struct net_device *netdev;
  	struct br2684_dev *brdev;
  	struct atm_newif_br2684 ni;
  
  	DPRINTK("br2684_create
  ");
  
  	if (copy_from_user(&ni, arg, sizeof ni)) {
  		return -EFAULT;
  	}
  	if (ni.media != BR2684_MEDIA_ETHERNET || ni.mtu != 1500) {
  		return -EINVAL;
  	}
  
  	netdev = alloc_netdev(sizeof(struct br2684_dev),
  			      ni.ifname[0] ? ni.ifname : "nas%d",
  			      br2684_setup);
  	if (!netdev)
  		return -ENOMEM;
  
  	brdev = BRPRIV(netdev);
  
  	DPRINTK("registered netdev %s
  ", netdev->name);
  	/* open, stop, do_ioctl ? */
  	err = register_netdev(netdev);
  	if (err < 0) {
  		printk(KERN_ERR "br2684_create: register_netdev failed
  ");
  		free_netdev(netdev);
  		return err;
  	}
  
  	write_lock_irq(&devs_lock);
  	brdev->number = list_empty(&br2684_devs) ? 1 :
  	    BRPRIV(list_entry_brdev(br2684_devs.prev))->number + 1;
  	list_add_tail(&brdev->br2684_devs, &br2684_devs);
  	write_unlock_irq(&devs_lock);
  	return 0;
  }
  
  /*
   * This handles ioctls actually performed on our vcc - we must return
   * -ENOIOCTLCMD for any unrecognized ioctl
   */
  static int br2684_ioctl(struct socket *sock, unsigned int cmd,
  	unsigned long arg)
  {
  	struct atm_vcc *atmvcc = ATM_SD(sock);
  	void __user *argp = (void __user *)arg;
  
  	int err;
  	switch(cmd) {
  	case ATM_SETBACKEND:
  	case ATM_NEWBACKENDIF: {
  		atm_backend_t b;
  		err = get_user(b, (atm_backend_t __user *) argp);
  		if (err)
  			return -EFAULT;
  		if (b != ATM_BACKEND_BR2684)
  			return -ENOIOCTLCMD;
  		if (!capable(CAP_NET_ADMIN))
  			return -EPERM;
  		if (cmd == ATM_SETBACKEND)
  			return br2684_regvcc(atmvcc, argp);
  		else
  			return br2684_create(argp);
  		}
  #ifdef CONFIG_ATM_BR2684_IPFILTER
  	case BR2684_SETFILT:
  		if (atmvcc->push != br2684_push)
  			return -ENOIOCTLCMD;
  		if (!capable(CAP_NET_ADMIN))
  			return -EPERM;
  		err = br2684_setfilt(atmvcc, argp);
  		return err;
  #endif /* CONFIG_ATM_BR2684_IPFILTER */
  	}
  	return -ENOIOCTLCMD;
  }
  
  static struct atm_ioctl br2684_ioctl_ops = {
  	.owner	= THIS_MODULE,
  	.ioctl	= br2684_ioctl,
  };
  
  
  #ifdef CONFIG_PROC_FS
  static void *br2684_seq_start(struct seq_file *seq, loff_t *pos)
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
711
  	read_lock(&devs_lock);
9af97186f   Pavel Emelianov   [ATM] br2684: Use...
712
  	return seq_list_start(&br2684_devs, *pos);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
713
714
715
716
  }
  
  static void *br2684_seq_next(struct seq_file *seq, void *v, loff_t *pos)
  {
9af97186f   Pavel Emelianov   [ATM] br2684: Use...
717
  	return seq_list_next(v, &br2684_devs, pos);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
718
719
720
721
722
723
724
725
726
  }
  
  static void br2684_seq_stop(struct seq_file *seq, void *v)
  {
  	read_unlock(&devs_lock);
  }
  
  static int br2684_seq_show(struct seq_file *seq, void *v)
  {
9af97186f   Pavel Emelianov   [ATM] br2684: Use...
727
728
  	const struct br2684_dev *brdev = list_entry(v, struct br2684_dev,
  			br2684_devs);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
  	const struct net_device *net_dev = brdev->net_dev;
  	const struct br2684_vcc *brvcc;
  
  	seq_printf(seq, "dev %.16s: num=%d, mac=%02X:%02X:"
  		       "%02X:%02X:%02X:%02X (%s)
  ", net_dev->name,
  		       brdev->number,
  		       net_dev->dev_addr[0],
  		       net_dev->dev_addr[1],
  		       net_dev->dev_addr[2],
  		       net_dev->dev_addr[3],
  		       net_dev->dev_addr[4],
  		       net_dev->dev_addr[5],
  		       brdev->mac_was_set ? "set" : "auto");
  
  	list_for_each_entry(brvcc, &brdev->brvccs, brvccs) {
  		seq_printf(seq, "  vcc %d.%d.%d: encaps=%s"
  #ifndef FASTER_VERSION
  				    ", failed copies %u/%u"
  #endif /* FASTER_VERSION */
  				    "
  ", brvcc->atmvcc->dev->number,
  				    brvcc->atmvcc->vpi, brvcc->atmvcc->vci,
  				    (brvcc->encaps == e_llc) ? "LLC" : "VC"
  #ifndef FASTER_VERSION
  				    , brvcc->copies_failed
  				    , brvcc->copies_needed
  #endif /* FASTER_VERSION */
  				    );
  #ifdef CONFIG_ATM_BR2684_IPFILTER
  #define b1(var, byte)	((u8 *) &brvcc->filter.var)[byte]
  #define bs(var)		b1(var, 0), b1(var, 1), b1(var, 2), b1(var, 3)
  			if (brvcc->filter.netmask != 0)
  				seq_printf(seq, "    filter=%d.%d.%d.%d/"
  						"%d.%d.%d.%d
  ",
  						bs(prefix), bs(netmask));
  #undef bs
  #undef b1
  #endif /* CONFIG_ATM_BR2684_IPFILTER */
  	}
  	return 0;
  }
56b3d975b   Philippe De Muyter   [NET]: Make all i...
772
  static const struct seq_operations br2684_seq_ops = {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
773
774
775
776
777
778
779
780
781
782
  	.start = br2684_seq_start,
  	.next  = br2684_seq_next,
  	.stop  = br2684_seq_stop,
  	.show  = br2684_seq_show,
  };
  
  static int br2684_proc_open(struct inode *inode, struct file *file)
  {
  	return seq_open(file, &br2684_seq_ops);
  }
9a32144e9   Arjan van de Ven   [PATCH] mark stru...
783
  static const struct file_operations br2684_proc_ops = {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
  	.owner   = THIS_MODULE,
  	.open    = br2684_proc_open,
  	.read    = seq_read,
  	.llseek  = seq_lseek,
  	.release = seq_release,
  };
  
  extern struct proc_dir_entry *atm_proc_root;	/* from proc.c */
  #endif
  
  static int __init br2684_init(void)
  {
  #ifdef CONFIG_PROC_FS
  	struct proc_dir_entry *p;
  	if ((p = create_proc_entry("br2684", 0, atm_proc_root)) == NULL)
  		return -ENOMEM;
  	p->proc_fops = &br2684_proc_ops;
  #endif
  	register_atm_ioctl(&br2684_ioctl_ops);
  	return 0;
  }
  
  static void __exit br2684_exit(void)
  {
  	struct net_device *net_dev;
  	struct br2684_dev *brdev;
  	struct br2684_vcc *brvcc;
  	deregister_atm_ioctl(&br2684_ioctl_ops);
  
  #ifdef CONFIG_PROC_FS
  	remove_proc_entry("br2684", atm_proc_root);
  #endif
  
  	while (!list_empty(&br2684_devs)) {
  		net_dev = list_entry_brdev(br2684_devs.next);
  		brdev = BRPRIV(net_dev);
  		while (!list_empty(&brdev->brvccs)) {
  			brvcc = list_entry_brvcc(brdev->brvccs.next);
  			br2684_close_vcc(brvcc);
  		}
  
  		list_del(&brdev->br2684_devs);
  		unregister_netdev(net_dev);
  		free_netdev(net_dev);
  	}
  }
  
  module_init(br2684_init);
  module_exit(br2684_exit);
  
  MODULE_AUTHOR("Marcell GAL");
  MODULE_DESCRIPTION("RFC2684 bridged protocols over ATM/AAL5");
  MODULE_LICENSE("GPL");