blob: 4e3f127e4003612106c3b1fa10aab971bd88e69b [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/* $Id: isdn_ppp.c,v 1.1.2.3 2004/02/10 01:07:13 keil Exp $
2 *
3 * Linux ISDN subsystem, functions for synchronous PPP (linklevel).
4 *
5 * Copyright 1995,96 by Michael Hipp (Michael.Hipp@student.uni-tuebingen.de)
6 *
7 * This software may be used and distributed according to the terms
8 * of the GNU General Public License, incorporated herein by reference.
9 *
10 */
11
Linus Torvalds1da177e2005-04-16 15:20:36 -070012#include <linux/isdn.h>
13#include <linux/poll.h>
14#include <linux/ppp-comp.h>
15#ifdef CONFIG_IPPP_FILTER
16#include <linux/filter.h>
17#endif
18
19#include "isdn_common.h"
20#include "isdn_ppp.h"
21#include "isdn_net.h"
22
23#ifndef PPP_IPX
24#define PPP_IPX 0x002b
25#endif
26
27/* Prototypes */
28static int isdn_ppp_fill_rq(unsigned char *buf, int len, int proto, int slot);
29static int isdn_ppp_closewait(int slot);
30static void isdn_ppp_push_higher(isdn_net_dev * net_dev, isdn_net_local * lp,
31 struct sk_buff *skb, int proto);
32static int isdn_ppp_if_get_unit(char *namebuf);
33static int isdn_ppp_set_compressor(struct ippp_struct *is,struct isdn_ppp_comp_data *);
34static struct sk_buff *isdn_ppp_decompress(struct sk_buff *,
35 struct ippp_struct *,struct ippp_struct *,int *proto);
36static void isdn_ppp_receive_ccp(isdn_net_dev * net_dev, isdn_net_local * lp,
37 struct sk_buff *skb,int proto);
38static struct sk_buff *isdn_ppp_compress(struct sk_buff *skb_in,int *proto,
39 struct ippp_struct *is,struct ippp_struct *master,int type);
40static void isdn_ppp_send_ccp(isdn_net_dev *net_dev, isdn_net_local *lp,
41 struct sk_buff *skb);
42
43/* New CCP stuff */
44static void isdn_ppp_ccp_kickup(struct ippp_struct *is);
45static void isdn_ppp_ccp_xmit_reset(struct ippp_struct *is, int proto,
46 unsigned char code, unsigned char id,
47 unsigned char *data, int len);
48static struct ippp_ccp_reset *isdn_ppp_ccp_reset_alloc(struct ippp_struct *is);
49static void isdn_ppp_ccp_reset_free(struct ippp_struct *is);
50static void isdn_ppp_ccp_reset_free_state(struct ippp_struct *is,
51 unsigned char id);
52static void isdn_ppp_ccp_timer_callback(unsigned long closure);
53static struct ippp_ccp_reset_state *isdn_ppp_ccp_reset_alloc_state(struct ippp_struct *is,
54 unsigned char id);
55static void isdn_ppp_ccp_reset_trans(struct ippp_struct *is,
56 struct isdn_ppp_resetparams *rp);
57static void isdn_ppp_ccp_reset_ack_rcvd(struct ippp_struct *is,
58 unsigned char id);
59
60
61
62#ifdef CONFIG_ISDN_MPP
63static ippp_bundle * isdn_ppp_bundle_arr = NULL;
64
65static int isdn_ppp_mp_bundle_array_init(void);
66static int isdn_ppp_mp_init( isdn_net_local * lp, ippp_bundle * add_to );
67static void isdn_ppp_mp_receive(isdn_net_dev * net_dev, isdn_net_local * lp,
68 struct sk_buff *skb);
69static void isdn_ppp_mp_cleanup( isdn_net_local * lp );
70
71static int isdn_ppp_bundle(struct ippp_struct *, int unit);
72#endif /* CONFIG_ISDN_MPP */
73
74char *isdn_ppp_revision = "$Revision: 1.1.2.3 $";
75
76static struct ippp_struct *ippp_table[ISDN_MAX_CHANNELS];
77
78static struct isdn_ppp_compressor *ipc_head = NULL;
79
80/*
81 * frame log (debug)
82 */
83static void
84isdn_ppp_frame_log(char *info, char *data, int len, int maxlen,int unit,int slot)
85{
86 int cnt,
87 j,
88 i;
89 char buf[80];
90
91 if (len < maxlen)
92 maxlen = len;
93
94 for (i = 0, cnt = 0; cnt < maxlen; i++) {
95 for (j = 0; j < 16 && cnt < maxlen; j++, cnt++)
96 sprintf(buf + j * 3, "%02x ", (unsigned char) data[cnt]);
97 printk(KERN_DEBUG "[%d/%d].%s[%d]: %s\n",unit,slot, info, i, buf);
98 }
99}
100
101/*
102 * unbind isdn_net_local <=> ippp-device
103 * note: it can happen, that we hangup/free the master before the slaves
104 * in this case we bind another lp to the master device
105 */
106int
107isdn_ppp_free(isdn_net_local * lp)
108{
109 struct ippp_struct *is;
110
Eric Sesterhenn052bb882006-04-11 17:29:17 -0700111 if (lp->ppp_slot < 0 || lp->ppp_slot >= ISDN_MAX_CHANNELS) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700112 printk(KERN_ERR "%s: ppp_slot(%d) out of range\n",
113 __FUNCTION__, lp->ppp_slot);
114 return 0;
115 }
116
117#ifdef CONFIG_ISDN_MPP
118 spin_lock(&lp->netdev->pb->lock);
119#endif
120 isdn_net_rm_from_bundle(lp);
121#ifdef CONFIG_ISDN_MPP
122 if (lp->netdev->pb->ref_ct == 1) /* last link in queue? */
123 isdn_ppp_mp_cleanup(lp);
124
125 lp->netdev->pb->ref_ct--;
126 spin_unlock(&lp->netdev->pb->lock);
127#endif /* CONFIG_ISDN_MPP */
Eric Sesterhenn052bb882006-04-11 17:29:17 -0700128 if (lp->ppp_slot < 0 || lp->ppp_slot >= ISDN_MAX_CHANNELS) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700129 printk(KERN_ERR "%s: ppp_slot(%d) now invalid\n",
130 __FUNCTION__, lp->ppp_slot);
131 return 0;
132 }
133 is = ippp_table[lp->ppp_slot];
134 if ((is->state & IPPP_CONNECT))
135 isdn_ppp_closewait(lp->ppp_slot); /* force wakeup on ippp device */
136 else if (is->state & IPPP_ASSIGNED)
137 is->state = IPPP_OPEN; /* fallback to 'OPEN but not ASSIGNED' state */
138
139 if (is->debug & 0x1)
140 printk(KERN_DEBUG "isdn_ppp_free %d %lx %lx\n", lp->ppp_slot, (long) lp, (long) is->lp);
141
142 is->lp = NULL; /* link is down .. set lp to NULL */
143 lp->ppp_slot = -1; /* is this OK ?? */
144
145 return 0;
146}
147
148/*
149 * bind isdn_net_local <=> ippp-device
150 *
151 * This function is allways called with holding dev->lock so
152 * no additional lock is needed
153 */
154int
155isdn_ppp_bind(isdn_net_local * lp)
156{
157 int i;
158 int unit = 0;
159 struct ippp_struct *is;
160 int retval;
161
162 if (lp->pppbind < 0) { /* device bounded to ippp device ? */
163 isdn_net_dev *net_dev = dev->netdev;
164 char exclusive[ISDN_MAX_CHANNELS]; /* exclusive flags */
165 memset(exclusive, 0, ISDN_MAX_CHANNELS);
166 while (net_dev) { /* step through net devices to find exclusive minors */
167 isdn_net_local *lp = net_dev->local;
168 if (lp->pppbind >= 0)
169 exclusive[lp->pppbind] = 1;
170 net_dev = net_dev->next;
171 }
172 /*
173 * search a free device / slot
174 */
175 for (i = 0; i < ISDN_MAX_CHANNELS; i++) {
176 if (ippp_table[i]->state == IPPP_OPEN && !exclusive[ippp_table[i]->minor]) { /* OPEN, but not connected! */
177 break;
178 }
179 }
180 } else {
181 for (i = 0; i < ISDN_MAX_CHANNELS; i++) {
182 if (ippp_table[i]->minor == lp->pppbind &&
183 (ippp_table[i]->state & IPPP_OPEN) == IPPP_OPEN)
184 break;
185 }
186 }
187
188 if (i >= ISDN_MAX_CHANNELS) {
189 printk(KERN_WARNING "isdn_ppp_bind: Can't find a (free) connection to the ipppd daemon.\n");
190 retval = -1;
191 goto out;
192 }
193 unit = isdn_ppp_if_get_unit(lp->name); /* get unit number from interface name .. ugly! */
194 if (unit < 0) {
195 printk(KERN_ERR "isdn_ppp_bind: illegal interface name %s.\n", lp->name);
196 retval = -1;
197 goto out;
198 }
199
200 lp->ppp_slot = i;
201 is = ippp_table[i];
202 is->lp = lp;
203 is->unit = unit;
204 is->state = IPPP_OPEN | IPPP_ASSIGNED; /* assigned to a netdevice but not connected */
205#ifdef CONFIG_ISDN_MPP
206 retval = isdn_ppp_mp_init(lp, NULL);
207 if (retval < 0)
208 goto out;
209#endif /* CONFIG_ISDN_MPP */
210
211 retval = lp->ppp_slot;
212
213 out:
214 return retval;
215}
216
217/*
218 * kick the ipppd on the device
219 * (wakes up daemon after B-channel connect)
220 */
221
222void
223isdn_ppp_wakeup_daemon(isdn_net_local * lp)
224{
225 if (lp->ppp_slot < 0 || lp->ppp_slot >= ISDN_MAX_CHANNELS) {
226 printk(KERN_ERR "%s: ppp_slot(%d) out of range\n",
227 __FUNCTION__, lp->ppp_slot);
228 return;
229 }
230 ippp_table[lp->ppp_slot]->state = IPPP_OPEN | IPPP_CONNECT | IPPP_NOBLOCK;
231 wake_up_interruptible(&ippp_table[lp->ppp_slot]->wq);
232}
233
234/*
235 * there was a hangup on the netdevice
236 * force wakeup of the ippp device
237 * go into 'device waits for release' state
238 */
239static int
240isdn_ppp_closewait(int slot)
241{
242 struct ippp_struct *is;
243
244 if (slot < 0 || slot >= ISDN_MAX_CHANNELS) {
245 printk(KERN_ERR "%s: slot(%d) out of range\n",
246 __FUNCTION__, slot);
247 return 0;
248 }
249 is = ippp_table[slot];
250 if (is->state)
251 wake_up_interruptible(&is->wq);
252 is->state = IPPP_CLOSEWAIT;
253 return 1;
254}
255
256/*
257 * isdn_ppp_find_slot / isdn_ppp_free_slot
258 */
259
260static int
261isdn_ppp_get_slot(void)
262{
263 int i;
264 for (i = 0; i < ISDN_MAX_CHANNELS; i++) {
265 if (!ippp_table[i]->state)
266 return i;
267 }
268 return -1;
269}
270
271/*
272 * isdn_ppp_open
273 */
274
275int
276isdn_ppp_open(int min, struct file *file)
277{
278 int slot;
279 struct ippp_struct *is;
280
Eric Sesterhenn052bb882006-04-11 17:29:17 -0700281 if (min < 0 || min >= ISDN_MAX_CHANNELS)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700282 return -ENODEV;
283
284 slot = isdn_ppp_get_slot();
285 if (slot < 0) {
286 return -EBUSY;
287 }
288 is = file->private_data = ippp_table[slot];
289
290 printk(KERN_DEBUG "ippp, open, slot: %d, minor: %d, state: %04x\n",
291 slot, min, is->state);
292
293 /* compression stuff */
294 is->link_compressor = is->compressor = NULL;
295 is->link_decompressor = is->decompressor = NULL;
296 is->link_comp_stat = is->comp_stat = NULL;
297 is->link_decomp_stat = is->decomp_stat = NULL;
298 is->compflags = 0;
299
300 is->reset = isdn_ppp_ccp_reset_alloc(is);
301
302 is->lp = NULL;
303 is->mp_seqno = 0; /* MP sequence number */
304 is->pppcfg = 0; /* ppp configuration */
305 is->mpppcfg = 0; /* mppp configuration */
306 is->last_link_seqno = -1; /* MP: maybe set to Bundle-MIN, when joining a bundle ?? */
307 is->unit = -1; /* set, when we have our interface */
308 is->mru = 1524; /* MRU, default 1524 */
309 is->maxcid = 16; /* VJ: maxcid */
310 is->tk = current;
311 init_waitqueue_head(&is->wq);
312 is->first = is->rq + NUM_RCV_BUFFS - 1; /* receive queue */
313 is->last = is->rq;
314 is->minor = min;
315#ifdef CONFIG_ISDN_PPP_VJ
316 /*
317 * VJ header compression init
318 */
319 is->slcomp = slhc_init(16, 16); /* not necessary for 2. link in bundle */
320#endif
321#ifdef CONFIG_IPPP_FILTER
322 is->pass_filter = NULL;
323 is->active_filter = NULL;
324#endif
325 is->state = IPPP_OPEN;
326
327 return 0;
328}
329
330/*
331 * release ippp device
332 */
333void
334isdn_ppp_release(int min, struct file *file)
335{
336 int i;
337 struct ippp_struct *is;
338
339 if (min < 0 || min >= ISDN_MAX_CHANNELS)
340 return;
341 is = file->private_data;
342
343 if (!is) {
344 printk(KERN_ERR "%s: no file->private_data\n", __FUNCTION__);
345 return;
346 }
347 if (is->debug & 0x1)
348 printk(KERN_DEBUG "ippp: release, minor: %d %lx\n", min, (long) is->lp);
349
350 if (is->lp) { /* a lp address says: this link is still up */
351 isdn_net_dev *p = is->lp->netdev;
352
353 if (!p) {
354 printk(KERN_ERR "%s: no lp->netdev\n", __FUNCTION__);
355 return;
356 }
357 is->state &= ~IPPP_CONNECT; /* -> effect: no call of wakeup */
358 /*
359 * isdn_net_hangup() calls isdn_ppp_free()
360 * isdn_ppp_free() sets is->lp to NULL and lp->ppp_slot to -1
361 * removing the IPPP_CONNECT flag omits calling of isdn_ppp_wakeup_daemon()
362 */
363 isdn_net_hangup(&p->dev);
364 }
365 for (i = 0; i < NUM_RCV_BUFFS; i++) {
Jesper Juhl3c7208f2005-11-07 01:01:29 -0800366 kfree(is->rq[i].buf);
367 is->rq[i].buf = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700368 }
369 is->first = is->rq + NUM_RCV_BUFFS - 1; /* receive queue */
370 is->last = is->rq;
371
372#ifdef CONFIG_ISDN_PPP_VJ
373/* TODO: if this was the previous master: link the slcomp to the new master */
374 slhc_free(is->slcomp);
375 is->slcomp = NULL;
376#endif
377#ifdef CONFIG_IPPP_FILTER
Jesper Juhl3c7208f2005-11-07 01:01:29 -0800378 kfree(is->pass_filter);
379 is->pass_filter = NULL;
380 kfree(is->active_filter);
381 is->active_filter = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700382#endif
383
384/* TODO: if this was the previous master: link the stuff to the new master */
385 if(is->comp_stat)
386 is->compressor->free(is->comp_stat);
387 if(is->link_comp_stat)
388 is->link_compressor->free(is->link_comp_stat);
389 if(is->link_decomp_stat)
390 is->link_decompressor->free(is->link_decomp_stat);
391 if(is->decomp_stat)
392 is->decompressor->free(is->decomp_stat);
393 is->compressor = is->link_compressor = NULL;
394 is->decompressor = is->link_decompressor = NULL;
395 is->comp_stat = is->link_comp_stat = NULL;
396 is->decomp_stat = is->link_decomp_stat = NULL;
397
398 /* Clean up if necessary */
399 if(is->reset)
400 isdn_ppp_ccp_reset_free(is);
401
402 /* this slot is ready for new connections */
403 is->state = 0;
404}
405
406/*
407 * get_arg .. ioctl helper
408 */
409static int
410get_arg(void __user *b, void *val, int len)
411{
412 if (len <= 0)
413 len = sizeof(void *);
414 if (copy_from_user(val, b, len))
415 return -EFAULT;
416 return 0;
417}
418
419/*
420 * set arg .. ioctl helper
421 */
422static int
423set_arg(void __user *b, void *val,int len)
424{
425 if(len <= 0)
426 len = sizeof(void *);
427 if (copy_to_user(b, val, len))
428 return -EFAULT;
429 return 0;
430}
431
432static int get_filter(void __user *arg, struct sock_filter **p)
433{
434 struct sock_fprog uprog;
435 struct sock_filter *code = NULL;
436 int len, err;
437
438 if (copy_from_user(&uprog, arg, sizeof(uprog)))
439 return -EFAULT;
440
441 if (!uprog.len) {
442 *p = NULL;
443 return 0;
444 }
445
446 /* uprog.len is unsigned short, so no overflow here */
447 len = uprog.len * sizeof(struct sock_filter);
448 code = kmalloc(len, GFP_KERNEL);
449 if (code == NULL)
450 return -ENOMEM;
451
452 if (copy_from_user(code, uprog.filter, len)) {
453 kfree(code);
454 return -EFAULT;
455 }
456
457 err = sk_chk_filter(code, uprog.len);
458 if (err) {
459 kfree(code);
460 return err;
461 }
462
463 *p = code;
464 return uprog.len;
465}
466
467/*
468 * ippp device ioctl
469 */
470int
471isdn_ppp_ioctl(int min, struct file *file, unsigned int cmd, unsigned long arg)
472{
473 unsigned long val;
474 int r,i,j;
475 struct ippp_struct *is;
476 isdn_net_local *lp;
477 struct isdn_ppp_comp_data data;
478 void __user *argp = (void __user *)arg;
479
480 is = (struct ippp_struct *) file->private_data;
481 lp = is->lp;
482
483 if (is->debug & 0x1)
484 printk(KERN_DEBUG "isdn_ppp_ioctl: minor: %d cmd: %x state: %x\n", min, cmd, is->state);
485
486 if (!(is->state & IPPP_OPEN))
487 return -EINVAL;
488
489 switch (cmd) {
490 case PPPIOCBUNDLE:
491#ifdef CONFIG_ISDN_MPP
492 if (!(is->state & IPPP_CONNECT))
493 return -EINVAL;
494 if ((r = get_arg(argp, &val, sizeof(val) )))
495 return r;
496 printk(KERN_DEBUG "iPPP-bundle: minor: %d, slave unit: %d, master unit: %d\n",
497 (int) min, (int) is->unit, (int) val);
498 return isdn_ppp_bundle(is, val);
499#else
500 return -1;
501#endif
502 break;
503 case PPPIOCGUNIT: /* get ppp/isdn unit number */
504 if ((r = set_arg(argp, &is->unit, sizeof(is->unit) )))
505 return r;
506 break;
507 case PPPIOCGIFNAME:
508 if(!lp)
509 return -EINVAL;
510 if ((r = set_arg(argp, lp->name, strlen(lp->name))))
511 return r;
512 break;
513 case PPPIOCGMPFLAGS: /* get configuration flags */
514 if ((r = set_arg(argp, &is->mpppcfg, sizeof(is->mpppcfg) )))
515 return r;
516 break;
517 case PPPIOCSMPFLAGS: /* set configuration flags */
518 if ((r = get_arg(argp, &val, sizeof(val) )))
519 return r;
520 is->mpppcfg = val;
521 break;
522 case PPPIOCGFLAGS: /* get configuration flags */
523 if ((r = set_arg(argp, &is->pppcfg,sizeof(is->pppcfg) )))
524 return r;
525 break;
526 case PPPIOCSFLAGS: /* set configuration flags */
527 if ((r = get_arg(argp, &val, sizeof(val) ))) {
528 return r;
529 }
530 if (val & SC_ENABLE_IP && !(is->pppcfg & SC_ENABLE_IP) && (is->state & IPPP_CONNECT)) {
531 if (lp) {
532 /* OK .. we are ready to send buffers */
533 is->pppcfg = val; /* isdn_ppp_xmit test for SC_ENABLE_IP !!! */
534 netif_wake_queue(&lp->netdev->dev);
535 break;
536 }
537 }
538 is->pppcfg = val;
539 break;
540 case PPPIOCGIDLE: /* get idle time information */
541 if (lp) {
542 struct ppp_idle pidle;
543 pidle.xmit_idle = pidle.recv_idle = lp->huptimer;
544 if ((r = set_arg(argp, &pidle,sizeof(struct ppp_idle))))
545 return r;
546 }
547 break;
548 case PPPIOCSMRU: /* set receive unit size for PPP */
549 if ((r = get_arg(argp, &val, sizeof(val) )))
550 return r;
551 is->mru = val;
552 break;
553 case PPPIOCSMPMRU:
554 break;
555 case PPPIOCSMPMTU:
556 break;
557 case PPPIOCSMAXCID: /* set the maximum compression slot id */
558 if ((r = get_arg(argp, &val, sizeof(val) )))
559 return r;
560 val++;
561 if (is->maxcid != val) {
562#ifdef CONFIG_ISDN_PPP_VJ
563 struct slcompress *sltmp;
564#endif
565 if (is->debug & 0x1)
566 printk(KERN_DEBUG "ippp, ioctl: changed MAXCID to %ld\n", val);
567 is->maxcid = val;
568#ifdef CONFIG_ISDN_PPP_VJ
569 sltmp = slhc_init(16, val);
570 if (!sltmp) {
571 printk(KERN_ERR "ippp, can't realloc slhc struct\n");
572 return -ENOMEM;
573 }
574 if (is->slcomp)
575 slhc_free(is->slcomp);
576 is->slcomp = sltmp;
577#endif
578 }
579 break;
580 case PPPIOCGDEBUG:
581 if ((r = set_arg(argp, &is->debug, sizeof(is->debug) )))
582 return r;
583 break;
584 case PPPIOCSDEBUG:
585 if ((r = get_arg(argp, &val, sizeof(val) )))
586 return r;
587 is->debug = val;
588 break;
589 case PPPIOCGCOMPRESSORS:
590 {
591 unsigned long protos[8] = {0,};
592 struct isdn_ppp_compressor *ipc = ipc_head;
593 while(ipc) {
594 j = ipc->num / (sizeof(long)*8);
595 i = ipc->num % (sizeof(long)*8);
596 if(j < 8)
597 protos[j] |= (0x1<<i);
598 ipc = ipc->next;
599 }
600 if ((r = set_arg(argp,protos,8*sizeof(long) )))
601 return r;
602 }
603 break;
604 case PPPIOCSCOMPRESSOR:
605 if ((r = get_arg(argp, &data, sizeof(struct isdn_ppp_comp_data))))
606 return r;
607 return isdn_ppp_set_compressor(is, &data);
608 case PPPIOCGCALLINFO:
609 {
610 struct pppcallinfo pci;
611 memset((char *) &pci,0,sizeof(struct pppcallinfo));
612 if(lp)
613 {
614 strncpy(pci.local_num,lp->msn,63);
615 if(lp->dial) {
616 strncpy(pci.remote_num,lp->dial->num,63);
617 }
618 pci.charge_units = lp->charge;
619 if(lp->outgoing)
620 pci.calltype = CALLTYPE_OUTGOING;
621 else
622 pci.calltype = CALLTYPE_INCOMING;
623 if(lp->flags & ISDN_NET_CALLBACK)
624 pci.calltype |= CALLTYPE_CALLBACK;
625 }
626 return set_arg(argp,&pci,sizeof(struct pppcallinfo));
627 }
628#ifdef CONFIG_IPPP_FILTER
629 case PPPIOCSPASS:
630 {
631 struct sock_filter *code;
632 int len = get_filter(argp, &code);
633 if (len < 0)
634 return len;
635 kfree(is->pass_filter);
636 is->pass_filter = code;
637 is->pass_len = len;
638 break;
639 }
640 case PPPIOCSACTIVE:
641 {
642 struct sock_filter *code;
643 int len = get_filter(argp, &code);
644 if (len < 0)
645 return len;
646 kfree(is->active_filter);
647 is->active_filter = code;
648 is->active_len = len;
649 break;
650 }
651#endif /* CONFIG_IPPP_FILTER */
652 default:
653 break;
654 }
655 return 0;
656}
657
658unsigned int
659isdn_ppp_poll(struct file *file, poll_table * wait)
660{
661 u_int mask;
662 struct ippp_buf_queue *bf, *bl;
663 u_long flags;
664 struct ippp_struct *is;
665
666 is = file->private_data;
667
668 if (is->debug & 0x2)
669 printk(KERN_DEBUG "isdn_ppp_poll: minor: %d\n",
Josef Sipek4482dfa2006-12-08 02:37:13 -0800670 iminor(file->f_path.dentry->d_inode));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700671
672 /* just registers wait_queue hook. This doesn't really wait. */
673 poll_wait(file, &is->wq, wait);
674
675 if (!(is->state & IPPP_OPEN)) {
676 if(is->state == IPPP_CLOSEWAIT)
677 return POLLHUP;
678 printk(KERN_DEBUG "isdn_ppp: device not open\n");
679 return POLLERR;
680 }
681 /* we're always ready to send .. */
682 mask = POLLOUT | POLLWRNORM;
683
684 spin_lock_irqsave(&is->buflock, flags);
685 bl = is->last;
686 bf = is->first;
687 /*
688 * if IPPP_NOBLOCK is set we return even if we have nothing to read
689 */
690 if (bf->next != bl || (is->state & IPPP_NOBLOCK)) {
691 is->state &= ~IPPP_NOBLOCK;
692 mask |= POLLIN | POLLRDNORM;
693 }
694 spin_unlock_irqrestore(&is->buflock, flags);
695 return mask;
696}
697
698/*
699 * fill up isdn_ppp_read() queue ..
700 */
701
702static int
703isdn_ppp_fill_rq(unsigned char *buf, int len, int proto, int slot)
704{
705 struct ippp_buf_queue *bf, *bl;
706 u_long flags;
707 u_char *nbuf;
708 struct ippp_struct *is;
709
710 if (slot < 0 || slot >= ISDN_MAX_CHANNELS) {
711 printk(KERN_WARNING "ippp: illegal slot(%d).\n", slot);
712 return 0;
713 }
714 is = ippp_table[slot];
715
716 if (!(is->state & IPPP_CONNECT)) {
717 printk(KERN_DEBUG "ippp: device not activated.\n");
718 return 0;
719 }
Robert P. J. Day5cbded52006-12-13 00:35:56 -0800720 nbuf = kmalloc(len + 4, GFP_ATOMIC);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700721 if (!nbuf) {
722 printk(KERN_WARNING "ippp: Can't alloc buf\n");
723 return 0;
724 }
725 nbuf[0] = PPP_ALLSTATIONS;
726 nbuf[1] = PPP_UI;
727 nbuf[2] = proto >> 8;
728 nbuf[3] = proto & 0xff;
729 memcpy(nbuf + 4, buf, len);
730
731 spin_lock_irqsave(&is->buflock, flags);
732 bf = is->first;
733 bl = is->last;
734
735 if (bf == bl) {
736 printk(KERN_WARNING "ippp: Queue is full; discarding first buffer\n");
737 bf = bf->next;
738 kfree(bf->buf);
739 is->first = bf;
740 }
741 bl->buf = (char *) nbuf;
742 bl->len = len + 4;
743
744 is->last = bl->next;
745 spin_unlock_irqrestore(&is->buflock, flags);
746 wake_up_interruptible(&is->wq);
747 return len;
748}
749
750/*
751 * read() .. non-blocking: ipppd calls it only after select()
752 * reports, that there is data
753 */
754
755int
756isdn_ppp_read(int min, struct file *file, char __user *buf, int count)
757{
758 struct ippp_struct *is;
759 struct ippp_buf_queue *b;
760 u_long flags;
761 u_char *save_buf;
762
763 is = file->private_data;
764
765 if (!(is->state & IPPP_OPEN))
766 return 0;
767
768 if (!access_ok(VERIFY_WRITE, buf, count))
769 return -EFAULT;
770
771 spin_lock_irqsave(&is->buflock, flags);
772 b = is->first->next;
773 save_buf = b->buf;
774 if (!save_buf) {
775 spin_unlock_irqrestore(&is->buflock, flags);
776 return -EAGAIN;
777 }
778 if (b->len < count)
779 count = b->len;
780 b->buf = NULL;
781 is->first = b;
782
783 spin_unlock_irqrestore(&is->buflock, flags);
Jesper Juhlc41a24c2006-03-25 03:07:02 -0800784 if (copy_to_user(buf, save_buf, count))
785 count = -EFAULT;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700786 kfree(save_buf);
787
788 return count;
789}
790
791/*
792 * ipppd wanna write a packet to the card .. non-blocking
793 */
794
795int
796isdn_ppp_write(int min, struct file *file, const char __user *buf, int count)
797{
798 isdn_net_local *lp;
799 struct ippp_struct *is;
800 int proto;
801 unsigned char protobuf[4];
802
803 is = file->private_data;
804
805 if (!(is->state & IPPP_CONNECT))
806 return 0;
807
808 lp = is->lp;
809
810 /* -> push it directly to the lowlevel interface */
811
812 if (!lp)
813 printk(KERN_DEBUG "isdn_ppp_write: lp == NULL\n");
814 else {
815 /*
816 * Don't reset huptimer for
817 * LCP packets. (Echo requests).
818 */
819 if (copy_from_user(protobuf, buf, 4))
820 return -EFAULT;
821 proto = PPP_PROTOCOL(protobuf);
822 if (proto != PPP_LCP)
823 lp->huptimer = 0;
824
825 if (lp->isdn_device < 0 || lp->isdn_channel < 0)
826 return 0;
827
828 if ((dev->drv[lp->isdn_device]->flags & DRV_FLAG_RUNNING) &&
829 lp->dialstate == 0 &&
830 (lp->flags & ISDN_NET_CONNECTED)) {
831 unsigned short hl;
832 struct sk_buff *skb;
833 /*
834 * we need to reserve enought space in front of
835 * sk_buff. old call to dev_alloc_skb only reserved
836 * 16 bytes, now we are looking what the driver want
837 */
838 hl = dev->drv[lp->isdn_device]->interface->hl_hdrlen;
839 skb = alloc_skb(hl+count, GFP_ATOMIC);
840 if (!skb) {
841 printk(KERN_WARNING "isdn_ppp_write: out of memory!\n");
842 return count;
843 }
844 skb_reserve(skb, hl);
845 if (copy_from_user(skb_put(skb, count), buf, count))
846 {
847 kfree_skb(skb);
848 return -EFAULT;
849 }
850 if (is->debug & 0x40) {
851 printk(KERN_DEBUG "ppp xmit: len %d\n", (int) skb->len);
852 isdn_ppp_frame_log("xmit", skb->data, skb->len, 32,is->unit,lp->ppp_slot);
853 }
854
855 isdn_ppp_send_ccp(lp->netdev,lp,skb); /* keeps CCP/compression states in sync */
856
857 isdn_net_write_super(lp, skb);
858 }
859 }
860 return count;
861}
862
863/*
864 * init memory, structures etc.
865 */
866
867int
868isdn_ppp_init(void)
869{
870 int i,
871 j;
872
873#ifdef CONFIG_ISDN_MPP
874 if( isdn_ppp_mp_bundle_array_init() < 0 )
875 return -ENOMEM;
876#endif /* CONFIG_ISDN_MPP */
877
878 for (i = 0; i < ISDN_MAX_CHANNELS; i++) {
Burman Yan41f96932006-12-08 02:39:35 -0800879 if (!(ippp_table[i] = kzalloc(sizeof(struct ippp_struct), GFP_KERNEL))) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700880 printk(KERN_WARNING "isdn_ppp_init: Could not alloc ippp_table\n");
881 for (j = 0; j < i; j++)
882 kfree(ippp_table[j]);
883 return -1;
884 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700885 spin_lock_init(&ippp_table[i]->buflock);
886 ippp_table[i]->state = 0;
887 ippp_table[i]->first = ippp_table[i]->rq + NUM_RCV_BUFFS - 1;
888 ippp_table[i]->last = ippp_table[i]->rq;
889
890 for (j = 0; j < NUM_RCV_BUFFS; j++) {
891 ippp_table[i]->rq[j].buf = NULL;
892 ippp_table[i]->rq[j].last = ippp_table[i]->rq +
893 (NUM_RCV_BUFFS + j - 1) % NUM_RCV_BUFFS;
894 ippp_table[i]->rq[j].next = ippp_table[i]->rq + (j + 1) % NUM_RCV_BUFFS;
895 }
896 }
897 return 0;
898}
899
900void
901isdn_ppp_cleanup(void)
902{
903 int i;
904
905 for (i = 0; i < ISDN_MAX_CHANNELS; i++)
906 kfree(ippp_table[i]);
907
908#ifdef CONFIG_ISDN_MPP
Jesper Juhl3c7208f2005-11-07 01:01:29 -0800909 kfree(isdn_ppp_bundle_arr);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700910#endif /* CONFIG_ISDN_MPP */
911
912}
913
914/*
915 * check for address/control field and skip if allowed
916 * retval != 0 -> discard packet silently
917 */
918static int isdn_ppp_skip_ac(struct ippp_struct *is, struct sk_buff *skb)
919{
920 if (skb->len < 1)
921 return -1;
922
923 if (skb->data[0] == 0xff) {
924 if (skb->len < 2)
925 return -1;
926
927 if (skb->data[1] != 0x03)
928 return -1;
929
930 // skip address/control (AC) field
931 skb_pull(skb, 2);
932 } else {
933 if (is->pppcfg & SC_REJ_COMP_AC)
934 // if AC compression was not negotiated, but used, discard packet
935 return -1;
936 }
937 return 0;
938}
939
940/*
941 * get the PPP protocol header and pull skb
942 * retval < 0 -> discard packet silently
943 */
944static int isdn_ppp_strip_proto(struct sk_buff *skb)
945{
946 int proto;
947
948 if (skb->len < 1)
949 return -1;
950
951 if (skb->data[0] & 0x1) {
952 // protocol field is compressed
953 proto = skb->data[0];
954 skb_pull(skb, 1);
955 } else {
956 if (skb->len < 2)
957 return -1;
958 proto = ((int) skb->data[0] << 8) + skb->data[1];
959 skb_pull(skb, 2);
960 }
961 return proto;
962}
963
964
965/*
966 * handler for incoming packets on a syncPPP interface
967 */
968void isdn_ppp_receive(isdn_net_dev * net_dev, isdn_net_local * lp, struct sk_buff *skb)
969{
970 struct ippp_struct *is;
971 int slot;
972 int proto;
973
Eric Sesterhenn6dd44a72006-03-26 18:19:26 +0200974 BUG_ON(net_dev->local->master); // we're called with the master device always
Linus Torvalds1da177e2005-04-16 15:20:36 -0700975
976 slot = lp->ppp_slot;
977 if (slot < 0 || slot >= ISDN_MAX_CHANNELS) {
978 printk(KERN_ERR "isdn_ppp_receive: lp->ppp_slot(%d)\n",
979 lp->ppp_slot);
980 kfree_skb(skb);
981 return;
982 }
983 is = ippp_table[slot];
984
985 if (is->debug & 0x4) {
986 printk(KERN_DEBUG "ippp_receive: is:%08lx lp:%08lx slot:%d unit:%d len:%d\n",
987 (long)is,(long)lp,lp->ppp_slot,is->unit,(int) skb->len);
988 isdn_ppp_frame_log("receive", skb->data, skb->len, 32,is->unit,lp->ppp_slot);
989 }
990
991 if (isdn_ppp_skip_ac(is, skb) < 0) {
992 kfree_skb(skb);
993 return;
994 }
995 proto = isdn_ppp_strip_proto(skb);
996 if (proto < 0) {
997 kfree_skb(skb);
998 return;
999 }
1000
1001#ifdef CONFIG_ISDN_MPP
1002 if (is->compflags & SC_LINK_DECOMP_ON) {
1003 skb = isdn_ppp_decompress(skb, is, NULL, &proto);
1004 if (!skb) // decompression error
1005 return;
1006 }
1007
1008 if (!(is->mpppcfg & SC_REJ_MP_PROT)) { // we agreed to receive MPPP
1009 if (proto == PPP_MP) {
1010 isdn_ppp_mp_receive(net_dev, lp, skb);
1011 return;
1012 }
1013 }
1014#endif
1015 isdn_ppp_push_higher(net_dev, lp, skb, proto);
1016}
1017
1018/*
1019 * we receive a reassembled frame, MPPP has been taken care of before.
1020 * address/control and protocol have been stripped from the skb
1021 * note: net_dev has to be master net_dev
1022 */
1023static void
1024isdn_ppp_push_higher(isdn_net_dev * net_dev, isdn_net_local * lp, struct sk_buff *skb, int proto)
1025{
1026 struct net_device *dev = &net_dev->dev;
1027 struct ippp_struct *is, *mis;
1028 isdn_net_local *mlp = NULL;
1029 int slot;
1030
1031 slot = lp->ppp_slot;
1032 if (slot < 0 || slot >= ISDN_MAX_CHANNELS) {
1033 printk(KERN_ERR "isdn_ppp_push_higher: lp->ppp_slot(%d)\n",
1034 lp->ppp_slot);
1035 goto drop_packet;
1036 }
1037 is = ippp_table[slot];
1038
1039 if (lp->master) { // FIXME?
1040 mlp = (isdn_net_local *) lp->master->priv;
1041 slot = mlp->ppp_slot;
Eric Sesterhenn052bb882006-04-11 17:29:17 -07001042 if (slot < 0 || slot >= ISDN_MAX_CHANNELS) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001043 printk(KERN_ERR "isdn_ppp_push_higher: master->ppp_slot(%d)\n",
1044 lp->ppp_slot);
1045 goto drop_packet;
1046 }
1047 }
1048 mis = ippp_table[slot];
1049
1050 if (is->debug & 0x10) {
1051 printk(KERN_DEBUG "push, skb %d %04x\n", (int) skb->len, proto);
1052 isdn_ppp_frame_log("rpush", skb->data, skb->len, 32,is->unit,lp->ppp_slot);
1053 }
1054 if (mis->compflags & SC_DECOMP_ON) {
1055 skb = isdn_ppp_decompress(skb, is, mis, &proto);
1056 if (!skb) // decompression error
1057 return;
1058 }
1059 switch (proto) {
1060 case PPP_IPX: /* untested */
1061 if (is->debug & 0x20)
1062 printk(KERN_DEBUG "isdn_ppp: IPX\n");
1063 skb->protocol = htons(ETH_P_IPX);
1064 break;
1065 case PPP_IP:
1066 if (is->debug & 0x20)
1067 printk(KERN_DEBUG "isdn_ppp: IP\n");
1068 skb->protocol = htons(ETH_P_IP);
1069 break;
1070 case PPP_COMP:
1071 case PPP_COMPFRAG:
1072 printk(KERN_INFO "isdn_ppp: unexpected compressed frame dropped\n");
1073 goto drop_packet;
1074#ifdef CONFIG_ISDN_PPP_VJ
1075 case PPP_VJC_UNCOMP:
1076 if (is->debug & 0x20)
1077 printk(KERN_DEBUG "isdn_ppp: VJC_UNCOMP\n");
1078 if (net_dev->local->ppp_slot < 0) {
1079 printk(KERN_ERR "%s: net_dev->local->ppp_slot(%d) out of range\n",
1080 __FUNCTION__, net_dev->local->ppp_slot);
1081 goto drop_packet;
1082 }
1083 if (slhc_remember(ippp_table[net_dev->local->ppp_slot]->slcomp, skb->data, skb->len) <= 0) {
1084 printk(KERN_WARNING "isdn_ppp: received illegal VJC_UNCOMP frame!\n");
1085 goto drop_packet;
1086 }
1087 skb->protocol = htons(ETH_P_IP);
1088 break;
1089 case PPP_VJC_COMP:
1090 if (is->debug & 0x20)
1091 printk(KERN_DEBUG "isdn_ppp: VJC_COMP\n");
1092 {
1093 struct sk_buff *skb_old = skb;
1094 int pkt_len;
1095 skb = dev_alloc_skb(skb_old->len + 128);
1096
1097 if (!skb) {
1098 printk(KERN_WARNING "%s: Memory squeeze, dropping packet.\n", dev->name);
1099 skb = skb_old;
1100 goto drop_packet;
1101 }
1102 skb_put(skb, skb_old->len + 128);
1103 memcpy(skb->data, skb_old->data, skb_old->len);
1104 if (net_dev->local->ppp_slot < 0) {
1105 printk(KERN_ERR "%s: net_dev->local->ppp_slot(%d) out of range\n",
1106 __FUNCTION__, net_dev->local->ppp_slot);
1107 goto drop_packet;
1108 }
1109 pkt_len = slhc_uncompress(ippp_table[net_dev->local->ppp_slot]->slcomp,
1110 skb->data, skb_old->len);
1111 kfree_skb(skb_old);
1112 if (pkt_len < 0)
1113 goto drop_packet;
1114
1115 skb_trim(skb, pkt_len);
1116 skb->protocol = htons(ETH_P_IP);
1117 }
1118 break;
1119#endif
1120 case PPP_CCP:
1121 case PPP_CCPFRAG:
1122 isdn_ppp_receive_ccp(net_dev,lp,skb,proto);
1123 /* Dont pop up ResetReq/Ack stuff to the daemon any
1124 longer - the job is done already */
1125 if(skb->data[0] == CCP_RESETREQ ||
1126 skb->data[0] == CCP_RESETACK)
1127 break;
1128 /* fall through */
1129 default:
1130 isdn_ppp_fill_rq(skb->data, skb->len, proto, lp->ppp_slot); /* push data to pppd device */
1131 kfree_skb(skb);
1132 return;
1133 }
1134
1135#ifdef CONFIG_IPPP_FILTER
1136 /* check if the packet passes the pass and active filters
1137 * the filter instructions are constructed assuming
1138 * a four-byte PPP header on each packet (which is still present) */
1139 skb_push(skb, 4);
1140
1141 {
1142 u_int16_t *p = (u_int16_t *) skb->data;
1143
Karsten Keild8470b72005-04-21 08:30:30 -07001144 *p = 0; /* indicate inbound */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001145 }
1146
1147 if (is->pass_filter
1148 && sk_run_filter(skb, is->pass_filter, is->pass_len) == 0) {
1149 if (is->debug & 0x2)
1150 printk(KERN_DEBUG "IPPP: inbound frame filtered.\n");
1151 kfree_skb(skb);
1152 return;
1153 }
1154 if (!(is->active_filter
1155 && sk_run_filter(skb, is->active_filter,
1156 is->active_len) == 0)) {
1157 if (is->debug & 0x2)
1158 printk(KERN_DEBUG "IPPP: link-active filter: reseting huptimer.\n");
1159 lp->huptimer = 0;
1160 if (mlp)
1161 mlp->huptimer = 0;
1162 }
1163 skb_pull(skb, 4);
1164#else /* CONFIG_IPPP_FILTER */
1165 lp->huptimer = 0;
1166 if (mlp)
1167 mlp->huptimer = 0;
1168#endif /* CONFIG_IPPP_FILTER */
1169 skb->dev = dev;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001170 skb->mac.raw = skb->data;
1171 netif_rx(skb);
1172 /* net_dev->local->stats.rx_packets++; done in isdn_net.c */
1173 return;
1174
1175 drop_packet:
1176 net_dev->local->stats.rx_dropped++;
1177 kfree_skb(skb);
1178}
1179
1180/*
1181 * isdn_ppp_skb_push ..
1182 * checks whether we have enough space at the beginning of the skb
1183 * and allocs a new SKB if necessary
1184 */
1185static unsigned char *isdn_ppp_skb_push(struct sk_buff **skb_p,int len)
1186{
1187 struct sk_buff *skb = *skb_p;
1188
1189 if(skb_headroom(skb) < len) {
1190 struct sk_buff *nskb = skb_realloc_headroom(skb, len);
1191
1192 if (!nskb) {
1193 printk(KERN_ERR "isdn_ppp_skb_push: can't realloc headroom!\n");
1194 dev_kfree_skb(skb);
1195 return NULL;
1196 }
1197 printk(KERN_DEBUG "isdn_ppp_skb_push:under %d %d\n",skb_headroom(skb),len);
1198 dev_kfree_skb(skb);
1199 *skb_p = nskb;
1200 return skb_push(nskb, len);
1201 }
1202 return skb_push(skb,len);
1203}
1204
1205/*
1206 * send ppp frame .. we expect a PIDCOMPressable proto --
1207 * (here: currently always PPP_IP,PPP_VJC_COMP,PPP_VJC_UNCOMP)
1208 *
1209 * VJ compression may change skb pointer!!! .. requeue with old
1210 * skb isn't allowed!!
1211 */
1212
1213int
1214isdn_ppp_xmit(struct sk_buff *skb, struct net_device *netdev)
1215{
1216 isdn_net_local *lp,*mlp;
1217 isdn_net_dev *nd;
1218 unsigned int proto = PPP_IP; /* 0x21 */
1219 struct ippp_struct *ipt,*ipts;
1220 int slot, retval = 0;
1221
1222 mlp = (isdn_net_local *) (netdev->priv);
1223 nd = mlp->netdev; /* get master lp */
1224
1225 slot = mlp->ppp_slot;
1226 if (slot < 0 || slot >= ISDN_MAX_CHANNELS) {
1227 printk(KERN_ERR "isdn_ppp_xmit: lp->ppp_slot(%d)\n",
1228 mlp->ppp_slot);
1229 kfree_skb(skb);
1230 goto out;
1231 }
1232 ipts = ippp_table[slot];
1233
1234 if (!(ipts->pppcfg & SC_ENABLE_IP)) { /* PPP connected ? */
1235 if (ipts->debug & 0x1)
1236 printk(KERN_INFO "%s: IP frame delayed.\n", netdev->name);
1237 retval = 1;
1238 goto out;
1239 }
1240
1241 switch (ntohs(skb->protocol)) {
1242 case ETH_P_IP:
1243 proto = PPP_IP;
1244 break;
1245 case ETH_P_IPX:
1246 proto = PPP_IPX; /* untested */
1247 break;
1248 default:
1249 printk(KERN_ERR "isdn_ppp: skipped unsupported protocol: %#x.\n",
1250 skb->protocol);
1251 dev_kfree_skb(skb);
1252 goto out;
1253 }
1254
1255 lp = isdn_net_get_locked_lp(nd);
1256 if (!lp) {
1257 printk(KERN_WARNING "%s: all channels busy - requeuing!\n", netdev->name);
1258 retval = 1;
1259 goto out;
1260 }
1261 /* we have our lp locked from now on */
1262
1263 slot = lp->ppp_slot;
Eric Sesterhenn052bb882006-04-11 17:29:17 -07001264 if (slot < 0 || slot >= ISDN_MAX_CHANNELS) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001265 printk(KERN_ERR "isdn_ppp_xmit: lp->ppp_slot(%d)\n",
1266 lp->ppp_slot);
1267 kfree_skb(skb);
1268 goto unlock;
1269 }
1270 ipt = ippp_table[slot];
1271
1272 /*
1273 * after this line .. requeueing in the device queue is no longer allowed!!!
1274 */
1275
1276 /* Pull off the fake header we stuck on earlier to keep
1277 * the fragmentation code happy.
1278 */
1279 skb_pull(skb,IPPP_MAX_HEADER);
1280
1281#ifdef CONFIG_IPPP_FILTER
1282 /* check if we should pass this packet
1283 * the filter instructions are constructed assuming
1284 * a four-byte PPP header on each packet */
Karsten Keild8470b72005-04-21 08:30:30 -07001285 *skb_push(skb, 4) = 1; /* indicate outbound */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001286
1287 {
1288 u_int16_t *p = (u_int16_t *) skb->data;
1289
Karsten Keild8470b72005-04-21 08:30:30 -07001290 p++;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001291 *p = htons(proto);
1292 }
1293
1294 if (ipt->pass_filter
1295 && sk_run_filter(skb, ipt->pass_filter, ipt->pass_len) == 0) {
1296 if (ipt->debug & 0x4)
1297 printk(KERN_DEBUG "IPPP: outbound frame filtered.\n");
1298 kfree_skb(skb);
1299 goto unlock;
1300 }
1301 if (!(ipt->active_filter
1302 && sk_run_filter(skb, ipt->active_filter,
1303 ipt->active_len) == 0)) {
1304 if (ipt->debug & 0x4)
1305 printk(KERN_DEBUG "IPPP: link-active filter: reseting huptimer.\n");
1306 lp->huptimer = 0;
1307 }
1308 skb_pull(skb, 4);
1309#else /* CONFIG_IPPP_FILTER */
1310 lp->huptimer = 0;
1311#endif /* CONFIG_IPPP_FILTER */
1312
1313 if (ipt->debug & 0x4)
1314 printk(KERN_DEBUG "xmit skb, len %d\n", (int) skb->len);
1315 if (ipts->debug & 0x40)
1316 isdn_ppp_frame_log("xmit0", skb->data, skb->len, 32,ipts->unit,lp->ppp_slot);
1317
1318#ifdef CONFIG_ISDN_PPP_VJ
1319 if (proto == PPP_IP && ipts->pppcfg & SC_COMP_TCP) { /* ipts here? probably yes, but check this again */
1320 struct sk_buff *new_skb;
1321 unsigned short hl;
1322 /*
1323 * we need to reserve enought space in front of
1324 * sk_buff. old call to dev_alloc_skb only reserved
1325 * 16 bytes, now we are looking what the driver want.
1326 */
1327 hl = dev->drv[lp->isdn_device]->interface->hl_hdrlen + IPPP_MAX_HEADER;
1328 /*
1329 * Note: hl might still be insufficient because the method
1330 * above does not account for a possibible MPPP slave channel
1331 * which had larger HL header space requirements than the
1332 * master.
1333 */
1334 new_skb = alloc_skb(hl+skb->len, GFP_ATOMIC);
1335 if (new_skb) {
1336 u_char *buf;
1337 int pktlen;
1338
1339 skb_reserve(new_skb, hl);
1340 new_skb->dev = skb->dev;
1341 skb_put(new_skb, skb->len);
1342 buf = skb->data;
1343
1344 pktlen = slhc_compress(ipts->slcomp, skb->data, skb->len, new_skb->data,
1345 &buf, !(ipts->pppcfg & SC_NO_TCP_CCID));
1346
1347 if (buf != skb->data) {
1348 if (new_skb->data != buf)
1349 printk(KERN_ERR "isdn_ppp: FATAL error after slhc_compress!!\n");
1350 dev_kfree_skb(skb);
1351 skb = new_skb;
1352 } else {
1353 dev_kfree_skb(new_skb);
1354 }
1355
1356 skb_trim(skb, pktlen);
1357 if (skb->data[0] & SL_TYPE_COMPRESSED_TCP) { /* cslip? style -> PPP */
1358 proto = PPP_VJC_COMP;
1359 skb->data[0] ^= SL_TYPE_COMPRESSED_TCP;
1360 } else {
1361 if (skb->data[0] >= SL_TYPE_UNCOMPRESSED_TCP)
1362 proto = PPP_VJC_UNCOMP;
1363 skb->data[0] = (skb->data[0] & 0x0f) | 0x40;
1364 }
1365 }
1366 }
1367#endif
1368
1369 /*
1370 * normal (single link) or bundle compression
1371 */
1372 if(ipts->compflags & SC_COMP_ON) {
1373 /* We send compressed only if both down- und upstream
1374 compression is negotiated, that means, CCP is up */
1375 if(ipts->compflags & SC_DECOMP_ON) {
1376 skb = isdn_ppp_compress(skb,&proto,ipt,ipts,0);
1377 } else {
1378 printk(KERN_DEBUG "isdn_ppp: CCP not yet up - sending as-is\n");
1379 }
1380 }
1381
1382 if (ipt->debug & 0x24)
1383 printk(KERN_DEBUG "xmit2 skb, len %d, proto %04x\n", (int) skb->len, proto);
1384
1385#ifdef CONFIG_ISDN_MPP
1386 if (ipt->mpppcfg & SC_MP_PROT) {
1387 /* we get mp_seqno from static isdn_net_local */
1388 long mp_seqno = ipts->mp_seqno;
1389 ipts->mp_seqno++;
1390 if (ipt->mpppcfg & SC_OUT_SHORT_SEQ) {
1391 unsigned char *data = isdn_ppp_skb_push(&skb, 3);
1392 if(!data)
1393 goto unlock;
1394 mp_seqno &= 0xfff;
1395 data[0] = MP_BEGIN_FRAG | MP_END_FRAG | ((mp_seqno >> 8) & 0xf); /* (B)egin & (E)ndbit .. */
1396 data[1] = mp_seqno & 0xff;
1397 data[2] = proto; /* PID compression */
1398 } else {
1399 unsigned char *data = isdn_ppp_skb_push(&skb, 5);
1400 if(!data)
1401 goto unlock;
1402 data[0] = MP_BEGIN_FRAG | MP_END_FRAG; /* (B)egin & (E)ndbit .. */
1403 data[1] = (mp_seqno >> 16) & 0xff; /* sequence number: 24bit */
1404 data[2] = (mp_seqno >> 8) & 0xff;
1405 data[3] = (mp_seqno >> 0) & 0xff;
1406 data[4] = proto; /* PID compression */
1407 }
1408 proto = PPP_MP; /* MP Protocol, 0x003d */
1409 }
1410#endif
1411
1412 /*
1413 * 'link in bundle' compression ...
1414 */
1415 if(ipt->compflags & SC_LINK_COMP_ON)
1416 skb = isdn_ppp_compress(skb,&proto,ipt,ipts,1);
1417
1418 if( (ipt->pppcfg & SC_COMP_PROT) && (proto <= 0xff) ) {
1419 unsigned char *data = isdn_ppp_skb_push(&skb,1);
1420 if(!data)
1421 goto unlock;
1422 data[0] = proto & 0xff;
1423 }
1424 else {
1425 unsigned char *data = isdn_ppp_skb_push(&skb,2);
1426 if(!data)
1427 goto unlock;
1428 data[0] = (proto >> 8) & 0xff;
1429 data[1] = proto & 0xff;
1430 }
1431 if(!(ipt->pppcfg & SC_COMP_AC)) {
1432 unsigned char *data = isdn_ppp_skb_push(&skb,2);
1433 if(!data)
1434 goto unlock;
1435 data[0] = 0xff; /* All Stations */
1436 data[1] = 0x03; /* Unnumbered information */
1437 }
1438
1439 /* tx-stats are now updated via BSENT-callback */
1440
1441 if (ipts->debug & 0x40) {
1442 printk(KERN_DEBUG "skb xmit: len: %d\n", (int) skb->len);
1443 isdn_ppp_frame_log("xmit", skb->data, skb->len, 32,ipt->unit,lp->ppp_slot);
1444 }
1445
1446 isdn_net_writebuf_skb(lp, skb);
1447
1448 unlock:
1449 spin_unlock_bh(&lp->xmit_lock);
1450 out:
1451 return retval;
1452}
1453
1454#ifdef CONFIG_IPPP_FILTER
1455/*
1456 * check if this packet may trigger auto-dial.
1457 */
1458
1459int isdn_ppp_autodial_filter(struct sk_buff *skb, isdn_net_local *lp)
1460{
1461 struct ippp_struct *is = ippp_table[lp->ppp_slot];
1462 u_int16_t proto;
1463 int drop = 0;
1464
1465 switch (ntohs(skb->protocol)) {
1466 case ETH_P_IP:
1467 proto = PPP_IP;
1468 break;
1469 case ETH_P_IPX:
1470 proto = PPP_IPX;
1471 break;
1472 default:
1473 printk(KERN_ERR "isdn_ppp_autodial_filter: unsupported protocol 0x%x.\n",
1474 skb->protocol);
1475 return 1;
1476 }
1477
1478 /* the filter instructions are constructed assuming
1479 * a four-byte PPP header on each packet. we have to
1480 * temporarily remove part of the fake header stuck on
1481 * earlier.
1482 */
Karsten Keild8470b72005-04-21 08:30:30 -07001483 *skb_pull(skb, IPPP_MAX_HEADER - 4) = 1; /* indicate outbound */
Linus Torvalds1da177e2005-04-16 15:20:36 -07001484
1485 {
1486 u_int16_t *p = (u_int16_t *) skb->data;
1487
Karsten Keild8470b72005-04-21 08:30:30 -07001488 p++;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001489 *p = htons(proto);
1490 }
1491
1492 drop |= is->pass_filter
1493 && sk_run_filter(skb, is->pass_filter, is->pass_len) == 0;
1494 drop |= is->active_filter
1495 && sk_run_filter(skb, is->active_filter, is->active_len) == 0;
1496
1497 skb_push(skb, IPPP_MAX_HEADER - 4);
1498 return drop;
1499}
1500#endif
1501#ifdef CONFIG_ISDN_MPP
1502
1503/* this is _not_ rfc1990 header, but something we convert both short and long
1504 * headers to for convinience's sake:
1505 * byte 0 is flags as in rfc1990
1506 * bytes 1...4 is 24-bit seqence number converted to host byte order
1507 */
1508#define MP_HEADER_LEN 5
1509
1510#define MP_LONGSEQ_MASK 0x00ffffff
1511#define MP_SHORTSEQ_MASK 0x00000fff
1512#define MP_LONGSEQ_MAX MP_LONGSEQ_MASK
1513#define MP_SHORTSEQ_MAX MP_SHORTSEQ_MASK
1514#define MP_LONGSEQ_MAXBIT ((MP_LONGSEQ_MASK+1)>>1)
1515#define MP_SHORTSEQ_MAXBIT ((MP_SHORTSEQ_MASK+1)>>1)
1516
1517/* sequence-wrap safe comparisions (for long sequence)*/
1518#define MP_LT(a,b) ((a-b)&MP_LONGSEQ_MAXBIT)
1519#define MP_LE(a,b) !((b-a)&MP_LONGSEQ_MAXBIT)
1520#define MP_GT(a,b) ((b-a)&MP_LONGSEQ_MAXBIT)
1521#define MP_GE(a,b) !((a-b)&MP_LONGSEQ_MAXBIT)
1522
1523#define MP_SEQ(f) ((*(u32*)(f->data+1)))
1524#define MP_FLAGS(f) (f->data[0])
1525
1526static int isdn_ppp_mp_bundle_array_init(void)
1527{
1528 int i;
1529 int sz = ISDN_MAX_CHANNELS*sizeof(ippp_bundle);
Burman Yan41f96932006-12-08 02:39:35 -08001530 if( (isdn_ppp_bundle_arr = kzalloc(sz, GFP_KERNEL)) == NULL )
Linus Torvalds1da177e2005-04-16 15:20:36 -07001531 return -ENOMEM;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001532 for( i = 0; i < ISDN_MAX_CHANNELS; i++ )
1533 spin_lock_init(&isdn_ppp_bundle_arr[i].lock);
1534 return 0;
1535}
1536
1537static ippp_bundle * isdn_ppp_mp_bundle_alloc(void)
1538{
1539 int i;
1540 for( i = 0; i < ISDN_MAX_CHANNELS; i++ )
1541 if (isdn_ppp_bundle_arr[i].ref_ct <= 0)
1542 return (isdn_ppp_bundle_arr + i);
1543 return NULL;
1544}
1545
1546static int isdn_ppp_mp_init( isdn_net_local * lp, ippp_bundle * add_to )
1547{
1548 struct ippp_struct * is;
1549
1550 if (lp->ppp_slot < 0) {
1551 printk(KERN_ERR "%s: lp->ppp_slot(%d) out of range\n",
1552 __FUNCTION__, lp->ppp_slot);
1553 return(-EINVAL);
1554 }
1555
1556 is = ippp_table[lp->ppp_slot];
1557 if (add_to) {
1558 if( lp->netdev->pb )
1559 lp->netdev->pb->ref_ct--;
1560 lp->netdev->pb = add_to;
1561 } else { /* first link in a bundle */
1562 is->mp_seqno = 0;
1563 if ((lp->netdev->pb = isdn_ppp_mp_bundle_alloc()) == NULL)
1564 return -ENOMEM;
1565 lp->next = lp->last = lp; /* nobody else in a queue */
1566 lp->netdev->pb->frags = NULL;
1567 lp->netdev->pb->frames = 0;
1568 lp->netdev->pb->seq = UINT_MAX;
1569 }
1570 lp->netdev->pb->ref_ct++;
1571
1572 is->last_link_seqno = 0;
1573 return 0;
1574}
1575
1576static u32 isdn_ppp_mp_get_seq( int short_seq,
1577 struct sk_buff * skb, u32 last_seq );
1578static struct sk_buff * isdn_ppp_mp_discard( ippp_bundle * mp,
1579 struct sk_buff * from, struct sk_buff * to );
1580static void isdn_ppp_mp_reassembly( isdn_net_dev * net_dev, isdn_net_local * lp,
1581 struct sk_buff * from, struct sk_buff * to );
1582static void isdn_ppp_mp_free_skb( ippp_bundle * mp, struct sk_buff * skb );
1583static void isdn_ppp_mp_print_recv_pkt( int slot, struct sk_buff * skb );
1584
1585static void isdn_ppp_mp_receive(isdn_net_dev * net_dev, isdn_net_local * lp,
1586 struct sk_buff *skb)
1587{
1588 struct ippp_struct *is;
1589 isdn_net_local * lpq;
1590 ippp_bundle * mp;
1591 isdn_mppp_stats * stats;
1592 struct sk_buff * newfrag, * frag, * start, *nextf;
1593 u32 newseq, minseq, thisseq;
1594 unsigned long flags;
1595 int slot;
1596
1597 spin_lock_irqsave(&net_dev->pb->lock, flags);
1598 mp = net_dev->pb;
1599 stats = &mp->stats;
1600 slot = lp->ppp_slot;
Eric Sesterhenn052bb882006-04-11 17:29:17 -07001601 if (slot < 0 || slot >= ISDN_MAX_CHANNELS) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001602 printk(KERN_ERR "%s: lp->ppp_slot(%d)\n",
1603 __FUNCTION__, lp->ppp_slot);
1604 stats->frame_drops++;
1605 dev_kfree_skb(skb);
1606 spin_unlock_irqrestore(&mp->lock, flags);
1607 return;
1608 }
1609 is = ippp_table[slot];
1610 if( ++mp->frames > stats->max_queue_len )
1611 stats->max_queue_len = mp->frames;
1612
1613 if (is->debug & 0x8)
1614 isdn_ppp_mp_print_recv_pkt(lp->ppp_slot, skb);
1615
1616 newseq = isdn_ppp_mp_get_seq(is->mpppcfg & SC_IN_SHORT_SEQ,
1617 skb, is->last_link_seqno);
1618
1619
1620 /* if this packet seq # is less than last already processed one,
1621 * toss it right away, but check for sequence start case first
1622 */
1623 if( mp->seq > MP_LONGSEQ_MAX && (newseq & MP_LONGSEQ_MAXBIT) ) {
1624 mp->seq = newseq; /* the first packet: required for
1625 * rfc1990 non-compliant clients --
1626 * prevents constant packet toss */
1627 } else if( MP_LT(newseq, mp->seq) ) {
1628 stats->frame_drops++;
1629 isdn_ppp_mp_free_skb(mp, skb);
1630 spin_unlock_irqrestore(&mp->lock, flags);
1631 return;
1632 }
1633
1634 /* find the minimum received sequence number over all links */
1635 is->last_link_seqno = minseq = newseq;
1636 for (lpq = net_dev->queue;;) {
1637 slot = lpq->ppp_slot;
Eric Sesterhenn052bb882006-04-11 17:29:17 -07001638 if (slot < 0 || slot >= ISDN_MAX_CHANNELS) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001639 printk(KERN_ERR "%s: lpq->ppp_slot(%d)\n",
1640 __FUNCTION__, lpq->ppp_slot);
1641 } else {
1642 u32 lls = ippp_table[slot]->last_link_seqno;
1643 if (MP_LT(lls, minseq))
1644 minseq = lls;
1645 }
1646 if ((lpq = lpq->next) == net_dev->queue)
1647 break;
1648 }
1649 if (MP_LT(minseq, mp->seq))
1650 minseq = mp->seq; /* can't go beyond already processed
1651 * packets */
1652 newfrag = skb;
1653
1654 /* if this new fragment is before the first one, then enqueue it now. */
1655 if ((frag = mp->frags) == NULL || MP_LT(newseq, MP_SEQ(frag))) {
1656 newfrag->next = frag;
1657 mp->frags = frag = newfrag;
1658 newfrag = NULL;
1659 }
1660
1661 start = MP_FLAGS(frag) & MP_BEGIN_FRAG &&
1662 MP_SEQ(frag) == mp->seq ? frag : NULL;
1663
1664 /*
1665 * main fragment traversing loop
1666 *
1667 * try to accomplish several tasks:
1668 * - insert new fragment into the proper sequence slot (once that's done
1669 * newfrag will be set to NULL)
1670 * - reassemble any complete fragment sequence (non-null 'start'
1671 * indicates there is a continguous sequence present)
1672 * - discard any incomplete sequences that are below minseq -- due
1673 * to the fact that sender always increment sequence number, if there
1674 * is an incomplete sequence below minseq, no new fragments would
1675 * come to complete such sequence and it should be discarded
1676 *
1677 * loop completes when we accomplished the following tasks:
1678 * - new fragment is inserted in the proper sequence ('newfrag' is
1679 * set to NULL)
1680 * - we hit a gap in the sequence, so no reassembly/processing is
1681 * possible ('start' would be set to NULL)
1682 *
1683 * algorightm for this code is derived from code in the book
1684 * 'PPP Design And Debugging' by James Carlson (Addison-Wesley)
1685 */
1686 while (start != NULL || newfrag != NULL) {
1687
1688 thisseq = MP_SEQ(frag);
1689 nextf = frag->next;
1690
1691 /* drop any duplicate fragments */
1692 if (newfrag != NULL && thisseq == newseq) {
1693 isdn_ppp_mp_free_skb(mp, newfrag);
1694 newfrag = NULL;
1695 }
1696
1697 /* insert new fragment before next element if possible. */
1698 if (newfrag != NULL && (nextf == NULL ||
1699 MP_LT(newseq, MP_SEQ(nextf)))) {
1700 newfrag->next = nextf;
1701 frag->next = nextf = newfrag;
1702 newfrag = NULL;
1703 }
1704
1705 if (start != NULL) {
1706 /* check for misplaced start */
1707 if (start != frag && (MP_FLAGS(frag) & MP_BEGIN_FRAG)) {
1708 printk(KERN_WARNING"isdn_mppp(seq %d): new "
1709 "BEGIN flag with no prior END", thisseq);
1710 stats->seqerrs++;
1711 stats->frame_drops++;
1712 start = isdn_ppp_mp_discard(mp, start,frag);
1713 nextf = frag->next;
1714 }
1715 } else if (MP_LE(thisseq, minseq)) {
1716 if (MP_FLAGS(frag) & MP_BEGIN_FRAG)
1717 start = frag;
1718 else {
1719 if (MP_FLAGS(frag) & MP_END_FRAG)
1720 stats->frame_drops++;
1721 if( mp->frags == frag )
1722 mp->frags = nextf;
1723 isdn_ppp_mp_free_skb(mp, frag);
1724 frag = nextf;
1725 continue;
1726 }
1727 }
1728
1729 /* if start is non-null and we have end fragment, then
1730 * we have full reassembly sequence -- reassemble
1731 * and process packet now
1732 */
1733 if (start != NULL && (MP_FLAGS(frag) & MP_END_FRAG)) {
1734 minseq = mp->seq = (thisseq+1) & MP_LONGSEQ_MASK;
1735 /* Reassemble the packet then dispatch it */
1736 isdn_ppp_mp_reassembly(net_dev, lp, start, nextf);
1737
1738 start = NULL;
1739 frag = NULL;
1740
1741 mp->frags = nextf;
1742 }
1743
1744 /* check if need to update start pointer: if we just
1745 * reassembled the packet and sequence is contiguous
1746 * then next fragment should be the start of new reassembly
1747 * if sequence is contiguous, but we haven't reassembled yet,
1748 * keep going.
1749 * if sequence is not contiguous, either clear everyting
1750 * below low watermark and set start to the next frag or
1751 * clear start ptr.
1752 */
1753 if (nextf != NULL &&
1754 ((thisseq+1) & MP_LONGSEQ_MASK) == MP_SEQ(nextf)) {
1755 /* if we just reassembled and the next one is here,
1756 * then start another reassembly. */
1757
1758 if (frag == NULL) {
1759 if (MP_FLAGS(nextf) & MP_BEGIN_FRAG)
1760 start = nextf;
1761 else
1762 {
1763 printk(KERN_WARNING"isdn_mppp(seq %d):"
1764 " END flag with no following "
1765 "BEGIN", thisseq);
1766 stats->seqerrs++;
1767 }
1768 }
1769
1770 } else {
1771 if ( nextf != NULL && frag != NULL &&
1772 MP_LT(thisseq, minseq)) {
1773 /* we've got a break in the sequence
1774 * and we not at the end yet
1775 * and we did not just reassembled
1776 *(if we did, there wouldn't be anything before)
1777 * and we below the low watermark
1778 * discard all the frames below low watermark
1779 * and start over */
1780 stats->frame_drops++;
1781 mp->frags = isdn_ppp_mp_discard(mp,start,nextf);
1782 }
1783 /* break in the sequence, no reassembly */
1784 start = NULL;
1785 }
1786
1787 frag = nextf;
1788 } /* while -- main loop */
1789
1790 if (mp->frags == NULL)
1791 mp->frags = frag;
1792
1793 /* rather straighforward way to deal with (not very) possible
1794 * queue overflow */
1795 if (mp->frames > MP_MAX_QUEUE_LEN) {
1796 stats->overflows++;
1797 while (mp->frames > MP_MAX_QUEUE_LEN) {
1798 frag = mp->frags->next;
1799 isdn_ppp_mp_free_skb(mp, mp->frags);
1800 mp->frags = frag;
1801 }
1802 }
1803 spin_unlock_irqrestore(&mp->lock, flags);
1804}
1805
1806static void isdn_ppp_mp_cleanup( isdn_net_local * lp )
1807{
1808 struct sk_buff * frag = lp->netdev->pb->frags;
1809 struct sk_buff * nextfrag;
1810 while( frag ) {
1811 nextfrag = frag->next;
1812 isdn_ppp_mp_free_skb(lp->netdev->pb, frag);
1813 frag = nextfrag;
1814 }
1815 lp->netdev->pb->frags = NULL;
1816}
1817
1818static u32 isdn_ppp_mp_get_seq( int short_seq,
1819 struct sk_buff * skb, u32 last_seq )
1820{
1821 u32 seq;
1822 int flags = skb->data[0] & (MP_BEGIN_FRAG | MP_END_FRAG);
1823
1824 if( !short_seq )
1825 {
1826 seq = ntohl(*(u32*)skb->data) & MP_LONGSEQ_MASK;
1827 skb_push(skb,1);
1828 }
1829 else
1830 {
1831 /* convert 12-bit short seq number to 24-bit long one
1832 */
1833 seq = ntohs(*(u16*)skb->data) & MP_SHORTSEQ_MASK;
1834
1835 /* check for seqence wrap */
1836 if( !(seq & MP_SHORTSEQ_MAXBIT) &&
1837 (last_seq & MP_SHORTSEQ_MAXBIT) &&
1838 (unsigned long)last_seq <= MP_LONGSEQ_MAX )
1839 seq |= (last_seq + MP_SHORTSEQ_MAX+1) &
1840 (~MP_SHORTSEQ_MASK & MP_LONGSEQ_MASK);
1841 else
1842 seq |= last_seq & (~MP_SHORTSEQ_MASK & MP_LONGSEQ_MASK);
1843
1844 skb_push(skb, 3); /* put converted seqence back in skb */
1845 }
1846 *(u32*)(skb->data+1) = seq; /* put seqence back in _host_ byte
1847 * order */
1848 skb->data[0] = flags; /* restore flags */
1849 return seq;
1850}
1851
1852struct sk_buff * isdn_ppp_mp_discard( ippp_bundle * mp,
1853 struct sk_buff * from, struct sk_buff * to )
1854{
1855 if( from )
1856 while (from != to) {
1857 struct sk_buff * next = from->next;
1858 isdn_ppp_mp_free_skb(mp, from);
1859 from = next;
1860 }
1861 return from;
1862}
1863
1864void isdn_ppp_mp_reassembly( isdn_net_dev * net_dev, isdn_net_local * lp,
1865 struct sk_buff * from, struct sk_buff * to )
1866{
1867 ippp_bundle * mp = net_dev->pb;
1868 int proto;
1869 struct sk_buff * skb;
1870 unsigned int tot_len;
1871
1872 if (lp->ppp_slot < 0 || lp->ppp_slot >= ISDN_MAX_CHANNELS) {
1873 printk(KERN_ERR "%s: lp->ppp_slot(%d) out of range\n",
1874 __FUNCTION__, lp->ppp_slot);
1875 return;
1876 }
1877 if( MP_FLAGS(from) == (MP_BEGIN_FRAG | MP_END_FRAG) ) {
1878 if( ippp_table[lp->ppp_slot]->debug & 0x40 )
1879 printk(KERN_DEBUG "isdn_mppp: reassembly: frame %d, "
1880 "len %d\n", MP_SEQ(from), from->len );
1881 skb = from;
1882 skb_pull(skb, MP_HEADER_LEN);
1883 mp->frames--;
1884 } else {
1885 struct sk_buff * frag;
1886 int n;
1887
1888 for(tot_len=n=0, frag=from; frag != to; frag=frag->next, n++)
1889 tot_len += frag->len - MP_HEADER_LEN;
1890
1891 if( ippp_table[lp->ppp_slot]->debug & 0x40 )
1892 printk(KERN_DEBUG"isdn_mppp: reassembling frames %d "
1893 "to %d, len %d\n", MP_SEQ(from),
1894 (MP_SEQ(from)+n-1) & MP_LONGSEQ_MASK, tot_len );
1895 if( (skb = dev_alloc_skb(tot_len)) == NULL ) {
1896 printk(KERN_ERR "isdn_mppp: cannot allocate sk buff "
1897 "of size %d\n", tot_len);
1898 isdn_ppp_mp_discard(mp, from, to);
1899 return;
1900 }
1901
1902 while( from != to ) {
1903 unsigned int len = from->len - MP_HEADER_LEN;
1904
1905 memcpy(skb_put(skb,len), from->data+MP_HEADER_LEN, len);
1906 frag = from->next;
1907 isdn_ppp_mp_free_skb(mp, from);
1908 from = frag;
1909 }
1910 }
1911 proto = isdn_ppp_strip_proto(skb);
1912 isdn_ppp_push_higher(net_dev, lp, skb, proto);
1913}
1914
1915static void isdn_ppp_mp_free_skb(ippp_bundle * mp, struct sk_buff * skb)
1916{
1917 dev_kfree_skb(skb);
1918 mp->frames--;
1919}
1920
1921static void isdn_ppp_mp_print_recv_pkt( int slot, struct sk_buff * skb )
1922{
1923 printk(KERN_DEBUG "mp_recv: %d/%d -> %02x %02x %02x %02x %02x %02x\n",
1924 slot, (int) skb->len,
1925 (int) skb->data[0], (int) skb->data[1], (int) skb->data[2],
1926 (int) skb->data[3], (int) skb->data[4], (int) skb->data[5]);
1927}
1928
1929static int
1930isdn_ppp_bundle(struct ippp_struct *is, int unit)
1931{
1932 char ifn[IFNAMSIZ + 1];
1933 isdn_net_dev *p;
1934 isdn_net_local *lp, *nlp;
1935 int rc;
1936 unsigned long flags;
1937
1938 sprintf(ifn, "ippp%d", unit);
1939 p = isdn_net_findif(ifn);
1940 if (!p) {
1941 printk(KERN_ERR "ippp_bundle: cannot find %s\n", ifn);
1942 return -EINVAL;
1943 }
1944
1945 spin_lock_irqsave(&p->pb->lock, flags);
1946
1947 nlp = is->lp;
1948 lp = p->queue;
1949 if( nlp->ppp_slot < 0 || nlp->ppp_slot >= ISDN_MAX_CHANNELS ||
1950 lp->ppp_slot < 0 || lp->ppp_slot >= ISDN_MAX_CHANNELS ) {
1951 printk(KERN_ERR "ippp_bundle: binding to invalid slot %d\n",
1952 nlp->ppp_slot < 0 || nlp->ppp_slot >= ISDN_MAX_CHANNELS ?
1953 nlp->ppp_slot : lp->ppp_slot );
1954 rc = -EINVAL;
1955 goto out;
1956 }
1957
1958 isdn_net_add_to_bundle(p, nlp);
1959
1960 ippp_table[nlp->ppp_slot]->unit = ippp_table[lp->ppp_slot]->unit;
1961
1962 /* maybe also SC_CCP stuff */
1963 ippp_table[nlp->ppp_slot]->pppcfg |= ippp_table[lp->ppp_slot]->pppcfg &
1964 (SC_ENABLE_IP | SC_NO_TCP_CCID | SC_REJ_COMP_TCP);
1965 ippp_table[nlp->ppp_slot]->mpppcfg |= ippp_table[lp->ppp_slot]->mpppcfg &
1966 (SC_MP_PROT | SC_REJ_MP_PROT | SC_OUT_SHORT_SEQ | SC_IN_SHORT_SEQ);
1967 rc = isdn_ppp_mp_init(nlp, p->pb);
1968out:
1969 spin_unlock_irqrestore(&p->pb->lock, flags);
1970 return rc;
1971}
1972
1973#endif /* CONFIG_ISDN_MPP */
1974
1975/*
1976 * network device ioctl handlers
1977 */
1978
1979static int
1980isdn_ppp_dev_ioctl_stats(int slot, struct ifreq *ifr, struct net_device *dev)
1981{
1982 struct ppp_stats __user *res = ifr->ifr_data;
1983 struct ppp_stats t;
1984 isdn_net_local *lp = (isdn_net_local *) dev->priv;
1985
1986 if (!access_ok(VERIFY_WRITE, res, sizeof(struct ppp_stats)))
1987 return -EFAULT;
1988
1989 /* build a temporary stat struct and copy it to user space */
1990
1991 memset(&t, 0, sizeof(struct ppp_stats));
1992 if (dev->flags & IFF_UP) {
1993 t.p.ppp_ipackets = lp->stats.rx_packets;
1994 t.p.ppp_ibytes = lp->stats.rx_bytes;
1995 t.p.ppp_ierrors = lp->stats.rx_errors;
1996 t.p.ppp_opackets = lp->stats.tx_packets;
1997 t.p.ppp_obytes = lp->stats.tx_bytes;
1998 t.p.ppp_oerrors = lp->stats.tx_errors;
1999#ifdef CONFIG_ISDN_PPP_VJ
2000 if (slot >= 0 && ippp_table[slot]->slcomp) {
2001 struct slcompress *slcomp = ippp_table[slot]->slcomp;
2002 t.vj.vjs_packets = slcomp->sls_o_compressed + slcomp->sls_o_uncompressed;
2003 t.vj.vjs_compressed = slcomp->sls_o_compressed;
2004 t.vj.vjs_searches = slcomp->sls_o_searches;
2005 t.vj.vjs_misses = slcomp->sls_o_misses;
2006 t.vj.vjs_errorin = slcomp->sls_i_error;
2007 t.vj.vjs_tossed = slcomp->sls_i_tossed;
2008 t.vj.vjs_uncompressedin = slcomp->sls_i_uncompressed;
2009 t.vj.vjs_compressedin = slcomp->sls_i_compressed;
2010 }
2011#endif
2012 }
2013 if (copy_to_user(res, &t, sizeof(struct ppp_stats)))
2014 return -EFAULT;
2015 return 0;
2016}
2017
2018int
2019isdn_ppp_dev_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd)
2020{
2021 int error=0;
2022 int len;
2023 isdn_net_local *lp = (isdn_net_local *) dev->priv;
2024
2025
2026 if (lp->p_encap != ISDN_NET_ENCAP_SYNCPPP)
2027 return -EINVAL;
2028
2029 switch (cmd) {
2030#define PPP_VERSION "2.3.7"
2031 case SIOCGPPPVER:
2032 len = strlen(PPP_VERSION) + 1;
2033 if (copy_to_user(ifr->ifr_data, PPP_VERSION, len))
2034 error = -EFAULT;
2035 break;
2036
2037 case SIOCGPPPSTATS:
2038 error = isdn_ppp_dev_ioctl_stats(lp->ppp_slot, ifr, dev);
2039 break;
2040 default:
2041 error = -EINVAL;
2042 break;
2043 }
2044 return error;
2045}
2046
2047static int
2048isdn_ppp_if_get_unit(char *name)
2049{
2050 int len,
2051 i,
2052 unit = 0,
2053 deci;
2054
2055 len = strlen(name);
2056
2057 if (strncmp("ippp", name, 4) || len > 8)
2058 return -1;
2059
2060 for (i = 0, deci = 1; i < len; i++, deci *= 10) {
2061 char a = name[len - i - 1];
2062 if (a >= '0' && a <= '9')
2063 unit += (a - '0') * deci;
2064 else
2065 break;
2066 }
2067 if (!i || len - i != 4)
2068 unit = -1;
2069
2070 return unit;
2071}
2072
2073
2074int
2075isdn_ppp_dial_slave(char *name)
2076{
2077#ifdef CONFIG_ISDN_MPP
2078 isdn_net_dev *ndev;
2079 isdn_net_local *lp;
2080 struct net_device *sdev;
2081
2082 if (!(ndev = isdn_net_findif(name)))
2083 return 1;
2084 lp = ndev->local;
2085 if (!(lp->flags & ISDN_NET_CONNECTED))
2086 return 5;
2087
2088 sdev = lp->slave;
2089 while (sdev) {
2090 isdn_net_local *mlp = (isdn_net_local *) sdev->priv;
2091 if (!(mlp->flags & ISDN_NET_CONNECTED))
2092 break;
2093 sdev = mlp->slave;
2094 }
2095 if (!sdev)
2096 return 2;
2097
2098 isdn_net_dial_req((isdn_net_local *) sdev->priv);
2099 return 0;
2100#else
2101 return -1;
2102#endif
2103}
2104
2105int
2106isdn_ppp_hangup_slave(char *name)
2107{
2108#ifdef CONFIG_ISDN_MPP
2109 isdn_net_dev *ndev;
2110 isdn_net_local *lp;
2111 struct net_device *sdev;
2112
2113 if (!(ndev = isdn_net_findif(name)))
2114 return 1;
2115 lp = ndev->local;
2116 if (!(lp->flags & ISDN_NET_CONNECTED))
2117 return 5;
2118
2119 sdev = lp->slave;
2120 while (sdev) {
2121 isdn_net_local *mlp = (isdn_net_local *) sdev->priv;
2122
2123 if (mlp->slave) { /* find last connected link in chain */
2124 isdn_net_local *nlp = (isdn_net_local *) mlp->slave->priv;
2125
2126 if (!(nlp->flags & ISDN_NET_CONNECTED))
2127 break;
2128 } else if (mlp->flags & ISDN_NET_CONNECTED)
2129 break;
2130
2131 sdev = mlp->slave;
2132 }
2133 if (!sdev)
2134 return 2;
2135
2136 isdn_net_hangup(sdev);
2137 return 0;
2138#else
2139 return -1;
2140#endif
2141}
2142
2143/*
2144 * PPP compression stuff
2145 */
2146
2147
2148/* Push an empty CCP Data Frame up to the daemon to wake it up and let it
2149 generate a CCP Reset-Request or tear down CCP altogether */
2150
2151static void isdn_ppp_ccp_kickup(struct ippp_struct *is)
2152{
2153 isdn_ppp_fill_rq(NULL, 0, PPP_COMP, is->lp->ppp_slot);
2154}
2155
2156/* In-kernel handling of CCP Reset-Request and Reset-Ack is necessary,
2157 but absolutely nontrivial. The most abstruse problem we are facing is
2158 that the generation, reception and all the handling of timeouts and
2159 resends including proper request id management should be entirely left
2160 to the (de)compressor, but indeed is not covered by the current API to
2161 the (de)compressor. The API is a prototype version from PPP where only
2162 some (de)compressors have yet been implemented and all of them are
2163 rather simple in their reset handling. Especially, their is only one
2164 outstanding ResetAck at a time with all of them and ResetReq/-Acks do
2165 not have parameters. For this very special case it was sufficient to
2166 just return an error code from the decompressor and have a single
2167 reset() entry to communicate all the necessary information between
2168 the framework and the (de)compressor. Bad enough, LZS is different
2169 (and any other compressor may be different, too). It has multiple
2170 histories (eventually) and needs to Reset each of them independently
2171 and thus uses multiple outstanding Acks and history numbers as an
2172 additional parameter to Reqs/Acks.
2173 All that makes it harder to port the reset state engine into the
2174 kernel because it is not just the same simple one as in (i)pppd but
2175 it must be able to pass additional parameters and have multiple out-
2176 standing Acks. We are trying to achieve the impossible by handling
2177 reset transactions independent by their id. The id MUST change when
2178 the data portion changes, thus any (de)compressor who uses more than
2179 one resettable state must provide and recognize individual ids for
2180 each individual reset transaction. The framework itself does _only_
2181 differentiate them by id, because it has no other semantics like the
2182 (de)compressor might.
2183 This looks like a major redesign of the interface would be nice,
2184 but I don't have an idea how to do it better. */
2185
2186/* Send a CCP Reset-Request or Reset-Ack directly from the kernel. This is
2187 getting that lengthy because there is no simple "send-this-frame-out"
2188 function above but every wrapper does a bit different. Hope I guess
2189 correct in this hack... */
2190
2191static void isdn_ppp_ccp_xmit_reset(struct ippp_struct *is, int proto,
2192 unsigned char code, unsigned char id,
2193 unsigned char *data, int len)
2194{
2195 struct sk_buff *skb;
2196 unsigned char *p;
2197 int hl;
2198 int cnt = 0;
2199 isdn_net_local *lp = is->lp;
2200
2201 /* Alloc large enough skb */
2202 hl = dev->drv[lp->isdn_device]->interface->hl_hdrlen;
2203 skb = alloc_skb(len + hl + 16,GFP_ATOMIC);
2204 if(!skb) {
2205 printk(KERN_WARNING
2206 "ippp: CCP cannot send reset - out of memory\n");
2207 return;
2208 }
2209 skb_reserve(skb, hl);
2210
2211 /* We may need to stuff an address and control field first */
2212 if(!(is->pppcfg & SC_COMP_AC)) {
2213 p = skb_put(skb, 2);
2214 *p++ = 0xff;
2215 *p++ = 0x03;
2216 }
2217
2218 /* Stuff proto, code, id and length */
2219 p = skb_put(skb, 6);
2220 *p++ = (proto >> 8);
2221 *p++ = (proto & 0xff);
2222 *p++ = code;
2223 *p++ = id;
2224 cnt = 4 + len;
2225 *p++ = (cnt >> 8);
2226 *p++ = (cnt & 0xff);
2227
2228 /* Now stuff remaining bytes */
2229 if(len) {
2230 p = skb_put(skb, len);
2231 memcpy(p, data, len);
2232 }
2233
2234 /* skb is now ready for xmit */
2235 printk(KERN_DEBUG "Sending CCP Frame:\n");
2236 isdn_ppp_frame_log("ccp-xmit", skb->data, skb->len, 32, is->unit,lp->ppp_slot);
2237
2238 isdn_net_write_super(lp, skb);
2239}
2240
2241/* Allocate the reset state vector */
2242static struct ippp_ccp_reset *isdn_ppp_ccp_reset_alloc(struct ippp_struct *is)
2243{
2244 struct ippp_ccp_reset *r;
Burman Yan41f96932006-12-08 02:39:35 -08002245 r = kzalloc(sizeof(struct ippp_ccp_reset), GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002246 if(!r) {
2247 printk(KERN_ERR "ippp_ccp: failed to allocate reset data"
2248 " structure - no mem\n");
2249 return NULL;
2250 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002251 printk(KERN_DEBUG "ippp_ccp: allocated reset data structure %p\n", r);
2252 is->reset = r;
2253 return r;
2254}
2255
2256/* Destroy the reset state vector. Kill all pending timers first. */
2257static void isdn_ppp_ccp_reset_free(struct ippp_struct *is)
2258{
2259 unsigned int id;
2260
2261 printk(KERN_DEBUG "ippp_ccp: freeing reset data structure %p\n",
2262 is->reset);
2263 for(id = 0; id < 256; id++) {
2264 if(is->reset->rs[id]) {
2265 isdn_ppp_ccp_reset_free_state(is, (unsigned char)id);
2266 }
2267 }
2268 kfree(is->reset);
2269 is->reset = NULL;
2270}
2271
2272/* Free a given state and clear everything up for later reallocation */
2273static void isdn_ppp_ccp_reset_free_state(struct ippp_struct *is,
2274 unsigned char id)
2275{
2276 struct ippp_ccp_reset_state *rs;
2277
2278 if(is->reset->rs[id]) {
2279 printk(KERN_DEBUG "ippp_ccp: freeing state for id %d\n", id);
2280 rs = is->reset->rs[id];
2281 /* Make sure the kernel will not call back later */
2282 if(rs->ta)
2283 del_timer(&rs->timer);
2284 is->reset->rs[id] = NULL;
2285 kfree(rs);
2286 } else {
2287 printk(KERN_WARNING "ippp_ccp: id %d is not allocated\n", id);
2288 }
2289}
2290
2291/* The timer callback function which is called when a ResetReq has timed out,
2292 aka has never been answered by a ResetAck */
2293static void isdn_ppp_ccp_timer_callback(unsigned long closure)
2294{
2295 struct ippp_ccp_reset_state *rs =
2296 (struct ippp_ccp_reset_state *)closure;
2297
2298 if(!rs) {
2299 printk(KERN_ERR "ippp_ccp: timer cb with zero closure.\n");
2300 return;
2301 }
2302 if(rs->ta && rs->state == CCPResetSentReq) {
2303 /* We are correct here */
2304 if(!rs->expra) {
2305 /* Hmm, there is no Ack really expected. We can clean
2306 up the state now, it will be reallocated if the
2307 decompressor insists on another reset */
2308 rs->ta = 0;
2309 isdn_ppp_ccp_reset_free_state(rs->is, rs->id);
2310 return;
2311 }
2312 printk(KERN_DEBUG "ippp_ccp: CCP Reset timed out for id %d\n",
2313 rs->id);
2314 /* Push it again */
2315 isdn_ppp_ccp_xmit_reset(rs->is, PPP_CCP, CCP_RESETREQ, rs->id,
2316 rs->data, rs->dlen);
2317 /* Restart timer */
2318 rs->timer.expires = jiffies + HZ*5;
2319 add_timer(&rs->timer);
2320 } else {
2321 printk(KERN_WARNING "ippp_ccp: timer cb in wrong state %d\n",
2322 rs->state);
2323 }
2324}
2325
2326/* Allocate a new reset transaction state */
2327static struct ippp_ccp_reset_state *isdn_ppp_ccp_reset_alloc_state(struct ippp_struct *is,
2328 unsigned char id)
2329{
2330 struct ippp_ccp_reset_state *rs;
2331 if(is->reset->rs[id]) {
2332 printk(KERN_WARNING "ippp_ccp: old state exists for id %d\n",
2333 id);
2334 return NULL;
2335 } else {
Burman Yan41f96932006-12-08 02:39:35 -08002336 rs = kzalloc(sizeof(struct ippp_ccp_reset_state), GFP_KERNEL);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002337 if(!rs)
2338 return NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002339 rs->state = CCPResetIdle;
2340 rs->is = is;
2341 rs->id = id;
Marcel Holtmanndab6df62006-12-21 23:06:24 +01002342 init_timer(&rs->timer);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002343 rs->timer.data = (unsigned long)rs;
2344 rs->timer.function = isdn_ppp_ccp_timer_callback;
2345 is->reset->rs[id] = rs;
2346 }
2347 return rs;
2348}
2349
2350
2351/* A decompressor wants a reset with a set of parameters - do what is
2352 necessary to fulfill it */
2353static void isdn_ppp_ccp_reset_trans(struct ippp_struct *is,
2354 struct isdn_ppp_resetparams *rp)
2355{
2356 struct ippp_ccp_reset_state *rs;
2357
2358 if(rp->valid) {
2359 /* The decompressor defines parameters by itself */
2360 if(rp->rsend) {
2361 /* And he wants us to send a request */
2362 if(!(rp->idval)) {
2363 printk(KERN_ERR "ippp_ccp: decompressor must"
2364 " specify reset id\n");
2365 return;
2366 }
2367 if(is->reset->rs[rp->id]) {
2368 /* There is already a transaction in existence
2369 for this id. May be still waiting for a
2370 Ack or may be wrong. */
2371 rs = is->reset->rs[rp->id];
2372 if(rs->state == CCPResetSentReq && rs->ta) {
2373 printk(KERN_DEBUG "ippp_ccp: reset"
2374 " trans still in progress"
2375 " for id %d\n", rp->id);
2376 } else {
2377 printk(KERN_WARNING "ippp_ccp: reset"
2378 " trans in wrong state %d for"
2379 " id %d\n", rs->state, rp->id);
2380 }
2381 } else {
2382 /* Ok, this is a new transaction */
2383 printk(KERN_DEBUG "ippp_ccp: new trans for id"
2384 " %d to be started\n", rp->id);
2385 rs = isdn_ppp_ccp_reset_alloc_state(is, rp->id);
2386 if(!rs) {
2387 printk(KERN_ERR "ippp_ccp: out of mem"
2388 " allocing ccp trans\n");
2389 return;
2390 }
2391 rs->state = CCPResetSentReq;
2392 rs->expra = rp->expra;
2393 if(rp->dtval) {
2394 rs->dlen = rp->dlen;
2395 memcpy(rs->data, rp->data, rp->dlen);
2396 }
2397 /* HACK TODO - add link comp here */
2398 isdn_ppp_ccp_xmit_reset(is, PPP_CCP,
2399 CCP_RESETREQ, rs->id,
2400 rs->data, rs->dlen);
2401 /* Start the timer */
2402 rs->timer.expires = jiffies + 5*HZ;
2403 add_timer(&rs->timer);
2404 rs->ta = 1;
2405 }
2406 } else {
2407 printk(KERN_DEBUG "ippp_ccp: no reset sent\n");
2408 }
2409 } else {
2410 /* The reset params are invalid. The decompressor does not
2411 care about them, so we just send the minimal requests
2412 and increase ids only when an Ack is received for a
2413 given id */
2414 if(is->reset->rs[is->reset->lastid]) {
2415 /* There is already a transaction in existence
2416 for this id. May be still waiting for a
2417 Ack or may be wrong. */
2418 rs = is->reset->rs[is->reset->lastid];
2419 if(rs->state == CCPResetSentReq && rs->ta) {
2420 printk(KERN_DEBUG "ippp_ccp: reset"
2421 " trans still in progress"
2422 " for id %d\n", rp->id);
2423 } else {
2424 printk(KERN_WARNING "ippp_ccp: reset"
2425 " trans in wrong state %d for"
2426 " id %d\n", rs->state, rp->id);
2427 }
2428 } else {
2429 printk(KERN_DEBUG "ippp_ccp: new trans for id"
2430 " %d to be started\n", is->reset->lastid);
2431 rs = isdn_ppp_ccp_reset_alloc_state(is,
2432 is->reset->lastid);
2433 if(!rs) {
2434 printk(KERN_ERR "ippp_ccp: out of mem"
2435 " allocing ccp trans\n");
2436 return;
2437 }
2438 rs->state = CCPResetSentReq;
2439 /* We always expect an Ack if the decompressor doesn't
2440 know better */
2441 rs->expra = 1;
2442 rs->dlen = 0;
2443 /* HACK TODO - add link comp here */
2444 isdn_ppp_ccp_xmit_reset(is, PPP_CCP, CCP_RESETREQ,
2445 rs->id, NULL, 0);
2446 /* Start the timer */
2447 rs->timer.expires = jiffies + 5*HZ;
2448 add_timer(&rs->timer);
2449 rs->ta = 1;
2450 }
2451 }
2452}
2453
2454/* An Ack was received for this id. This means we stop the timer and clean
2455 up the state prior to calling the decompressors reset routine. */
2456static void isdn_ppp_ccp_reset_ack_rcvd(struct ippp_struct *is,
2457 unsigned char id)
2458{
2459 struct ippp_ccp_reset_state *rs = is->reset->rs[id];
2460
2461 if(rs) {
2462 if(rs->ta && rs->state == CCPResetSentReq) {
2463 /* Great, we are correct */
2464 if(!rs->expra)
2465 printk(KERN_DEBUG "ippp_ccp: ResetAck received"
2466 " for id %d but not expected\n", id);
2467 } else {
2468 printk(KERN_INFO "ippp_ccp: ResetAck received out of"
2469 "sync for id %d\n", id);
2470 }
2471 if(rs->ta) {
2472 rs->ta = 0;
2473 del_timer(&rs->timer);
2474 }
2475 isdn_ppp_ccp_reset_free_state(is, id);
2476 } else {
2477 printk(KERN_INFO "ippp_ccp: ResetAck received for unknown id"
2478 " %d\n", id);
2479 }
2480 /* Make sure the simple reset stuff uses a new id next time */
2481 is->reset->lastid++;
2482}
2483
2484/*
2485 * decompress packet
2486 *
2487 * if master = 0, we're trying to uncompress an per-link compressed packet,
2488 * as opposed to an compressed reconstructed-from-MPPP packet.
2489 * proto is updated to protocol field of uncompressed packet.
2490 *
2491 * retval: decompressed packet,
2492 * same packet if uncompressed,
2493 * NULL if decompression error
2494 */
2495
2496static struct sk_buff *isdn_ppp_decompress(struct sk_buff *skb,struct ippp_struct *is,struct ippp_struct *master,
2497 int *proto)
2498{
2499 void *stat = NULL;
2500 struct isdn_ppp_compressor *ipc = NULL;
2501 struct sk_buff *skb_out;
2502 int len;
2503 struct ippp_struct *ri;
2504 struct isdn_ppp_resetparams rsparm;
2505 unsigned char rsdata[IPPP_RESET_MAXDATABYTES];
2506
2507 if(!master) {
2508 // per-link decompression
2509 stat = is->link_decomp_stat;
2510 ipc = is->link_decompressor;
2511 ri = is;
2512 } else {
2513 stat = master->decomp_stat;
2514 ipc = master->decompressor;
2515 ri = master;
2516 }
2517
2518 if (!ipc) {
2519 // no decompressor -> we can't decompress.
2520 printk(KERN_DEBUG "ippp: no decompressor defined!\n");
2521 return skb;
2522 }
Eric Sesterhenn6dd44a72006-03-26 18:19:26 +02002523 BUG_ON(!stat); // if we have a compressor, stat has been set as well
Linus Torvalds1da177e2005-04-16 15:20:36 -07002524
2525 if((master && *proto == PPP_COMP) || (!master && *proto == PPP_COMPFRAG) ) {
2526 // compressed packets are compressed by their protocol type
2527
2528 // Set up reset params for the decompressor
2529 memset(&rsparm, 0, sizeof(rsparm));
2530 rsparm.data = rsdata;
2531 rsparm.maxdlen = IPPP_RESET_MAXDATABYTES;
2532
2533 skb_out = dev_alloc_skb(is->mru + PPP_HDRLEN);
Jesper Juhlf6e2cdc2006-12-08 02:39:34 -08002534 if (!skb_out) {
2535 kfree_skb(skb);
2536 printk(KERN_ERR "ippp: decomp memory allocation failure\n");
2537 return NULL;
2538 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002539 len = ipc->decompress(stat, skb, skb_out, &rsparm);
2540 kfree_skb(skb);
2541 if (len <= 0) {
2542 switch(len) {
2543 case DECOMP_ERROR:
2544 printk(KERN_INFO "ippp: decomp wants reset %s params\n",
2545 rsparm.valid ? "with" : "without");
2546
2547 isdn_ppp_ccp_reset_trans(ri, &rsparm);
2548 break;
2549 case DECOMP_FATALERROR:
2550 ri->pppcfg |= SC_DC_FERROR;
2551 /* Kick ipppd to recognize the error */
2552 isdn_ppp_ccp_kickup(ri);
2553 break;
2554 }
2555 kfree_skb(skb_out);
2556 return NULL;
2557 }
2558 *proto = isdn_ppp_strip_proto(skb_out);
2559 if (*proto < 0) {
2560 kfree_skb(skb_out);
2561 return NULL;
2562 }
2563 return skb_out;
2564 } else {
2565 // uncompressed packets are fed through the decompressor to
2566 // update the decompressor state
2567 ipc->incomp(stat, skb, *proto);
2568 return skb;
2569 }
2570}
2571
2572/*
2573 * compress a frame
2574 * type=0: normal/bundle compression
2575 * =1: link compression
2576 * returns original skb if we haven't compressed the frame
2577 * and a new skb pointer if we've done it
2578 */
2579static struct sk_buff *isdn_ppp_compress(struct sk_buff *skb_in,int *proto,
2580 struct ippp_struct *is,struct ippp_struct *master,int type)
2581{
2582 int ret;
2583 int new_proto;
2584 struct isdn_ppp_compressor *compressor;
2585 void *stat;
2586 struct sk_buff *skb_out;
2587
2588 /* we do not compress control protocols */
2589 if(*proto < 0 || *proto > 0x3fff) {
2590 return skb_in;
2591 }
2592
2593 if(type) { /* type=1 => Link compression */
2594 return skb_in;
2595 }
2596 else {
2597 if(!master) {
2598 compressor = is->compressor;
2599 stat = is->comp_stat;
2600 }
2601 else {
2602 compressor = master->compressor;
2603 stat = master->comp_stat;
2604 }
2605 new_proto = PPP_COMP;
2606 }
2607
2608 if(!compressor) {
2609 printk(KERN_ERR "isdn_ppp: No compressor set!\n");
2610 return skb_in;
2611 }
2612 if(!stat) {
2613 printk(KERN_ERR "isdn_ppp: Compressor not initialized?\n");
2614 return skb_in;
2615 }
2616
2617 /* Allow for at least 150 % expansion (for now) */
2618 skb_out = alloc_skb(skb_in->len + skb_in->len/2 + 32 +
2619 skb_headroom(skb_in), GFP_ATOMIC);
2620 if(!skb_out)
2621 return skb_in;
2622 skb_reserve(skb_out, skb_headroom(skb_in));
2623
2624 ret = (compressor->compress)(stat,skb_in,skb_out,*proto);
2625 if(!ret) {
2626 dev_kfree_skb(skb_out);
2627 return skb_in;
2628 }
2629
2630 dev_kfree_skb(skb_in);
2631 *proto = new_proto;
2632 return skb_out;
2633}
2634
2635/*
2636 * we received a CCP frame ..
2637 * not a clean solution, but we MUST handle a few cases in the kernel
2638 */
2639static void isdn_ppp_receive_ccp(isdn_net_dev *net_dev, isdn_net_local *lp,
2640 struct sk_buff *skb,int proto)
2641{
2642 struct ippp_struct *is;
2643 struct ippp_struct *mis;
2644 int len;
2645 struct isdn_ppp_resetparams rsparm;
2646 unsigned char rsdata[IPPP_RESET_MAXDATABYTES];
2647
2648 printk(KERN_DEBUG "Received CCP frame from peer slot(%d)\n",
2649 lp->ppp_slot);
Eric Sesterhenn052bb882006-04-11 17:29:17 -07002650 if (lp->ppp_slot < 0 || lp->ppp_slot >= ISDN_MAX_CHANNELS) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002651 printk(KERN_ERR "%s: lp->ppp_slot(%d) out of range\n",
2652 __FUNCTION__, lp->ppp_slot);
2653 return;
2654 }
2655 is = ippp_table[lp->ppp_slot];
2656 isdn_ppp_frame_log("ccp-rcv", skb->data, skb->len, 32, is->unit,lp->ppp_slot);
2657
2658 if(lp->master) {
2659 int slot = ((isdn_net_local *) (lp->master->priv))->ppp_slot;
Eric Sesterhenn052bb882006-04-11 17:29:17 -07002660 if (slot < 0 || slot >= ISDN_MAX_CHANNELS) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002661 printk(KERN_ERR "%s: slot(%d) out of range\n",
2662 __FUNCTION__, slot);
2663 return;
2664 }
2665 mis = ippp_table[slot];
2666 } else
2667 mis = is;
2668
2669 switch(skb->data[0]) {
2670 case CCP_CONFREQ:
2671 if(is->debug & 0x10)
2672 printk(KERN_DEBUG "Disable compression here!\n");
2673 if(proto == PPP_CCP)
2674 mis->compflags &= ~SC_COMP_ON;
2675 else
2676 is->compflags &= ~SC_LINK_COMP_ON;
2677 break;
2678 case CCP_TERMREQ:
2679 case CCP_TERMACK:
2680 if(is->debug & 0x10)
2681 printk(KERN_DEBUG "Disable (de)compression here!\n");
2682 if(proto == PPP_CCP)
2683 mis->compflags &= ~(SC_DECOMP_ON|SC_COMP_ON);
2684 else
2685 is->compflags &= ~(SC_LINK_DECOMP_ON|SC_LINK_COMP_ON);
2686 break;
2687 case CCP_CONFACK:
2688 /* if we RECEIVE an ackowledge we enable the decompressor */
2689 if(is->debug & 0x10)
2690 printk(KERN_DEBUG "Enable decompression here!\n");
2691 if(proto == PPP_CCP) {
2692 if (!mis->decompressor)
2693 break;
2694 mis->compflags |= SC_DECOMP_ON;
2695 } else {
2696 if (!is->decompressor)
2697 break;
2698 is->compflags |= SC_LINK_DECOMP_ON;
2699 }
2700 break;
2701
2702 case CCP_RESETACK:
2703 printk(KERN_DEBUG "Received ResetAck from peer\n");
2704 len = (skb->data[2] << 8) | skb->data[3];
2705 len -= 4;
2706
2707 if(proto == PPP_CCP) {
2708 /* If a reset Ack was outstanding for this id, then
2709 clean up the state engine */
2710 isdn_ppp_ccp_reset_ack_rcvd(mis, skb->data[1]);
2711 if(mis->decompressor && mis->decomp_stat)
2712 mis->decompressor->
2713 reset(mis->decomp_stat,
2714 skb->data[0],
2715 skb->data[1],
2716 len ? &skb->data[4] : NULL,
2717 len, NULL);
2718 /* TODO: This is not easy to decide here */
2719 mis->compflags &= ~SC_DECOMP_DISCARD;
2720 }
2721 else {
2722 isdn_ppp_ccp_reset_ack_rcvd(is, skb->data[1]);
2723 if(is->link_decompressor && is->link_decomp_stat)
2724 is->link_decompressor->
2725 reset(is->link_decomp_stat,
2726 skb->data[0],
2727 skb->data[1],
2728 len ? &skb->data[4] : NULL,
2729 len, NULL);
2730 /* TODO: neither here */
2731 is->compflags &= ~SC_LINK_DECOMP_DISCARD;
2732 }
2733 break;
2734
2735 case CCP_RESETREQ:
2736 printk(KERN_DEBUG "Received ResetReq from peer\n");
2737 /* Receiving a ResetReq means we must reset our compressor */
2738 /* Set up reset params for the reset entry */
2739 memset(&rsparm, 0, sizeof(rsparm));
2740 rsparm.data = rsdata;
2741 rsparm.maxdlen = IPPP_RESET_MAXDATABYTES;
2742 /* Isolate data length */
2743 len = (skb->data[2] << 8) | skb->data[3];
2744 len -= 4;
2745 if(proto == PPP_CCP) {
2746 if(mis->compressor && mis->comp_stat)
2747 mis->compressor->
2748 reset(mis->comp_stat,
2749 skb->data[0],
2750 skb->data[1],
2751 len ? &skb->data[4] : NULL,
2752 len, &rsparm);
2753 }
2754 else {
2755 if(is->link_compressor && is->link_comp_stat)
2756 is->link_compressor->
2757 reset(is->link_comp_stat,
2758 skb->data[0],
2759 skb->data[1],
2760 len ? &skb->data[4] : NULL,
2761 len, &rsparm);
2762 }
2763 /* Ack the Req as specified by rsparm */
2764 if(rsparm.valid) {
2765 /* Compressor reset handler decided how to answer */
2766 if(rsparm.rsend) {
2767 /* We should send a Frame */
2768 isdn_ppp_ccp_xmit_reset(is, proto, CCP_RESETACK,
2769 rsparm.idval ? rsparm.id
2770 : skb->data[1],
2771 rsparm.dtval ?
2772 rsparm.data : NULL,
2773 rsparm.dtval ?
2774 rsparm.dlen : 0);
2775 } else {
2776 printk(KERN_DEBUG "ResetAck suppressed\n");
2777 }
2778 } else {
2779 /* We answer with a straight reflected Ack */
2780 isdn_ppp_ccp_xmit_reset(is, proto, CCP_RESETACK,
2781 skb->data[1],
2782 len ? &skb->data[4] : NULL,
2783 len);
2784 }
2785 break;
2786 }
2787}
2788
2789
2790/*
2791 * Daemon sends a CCP frame ...
2792 */
2793
2794/* TODO: Clean this up with new Reset semantics */
2795
2796/* I believe the CCP handling as-is is done wrong. Compressed frames
2797 * should only be sent/received after CCP reaches UP state, which means
2798 * both sides have sent CONF_ACK. Currently, we handle both directions
2799 * independently, which means we may accept compressed frames too early
2800 * (supposedly not a problem), but may also mean we send compressed frames
2801 * too early, which may turn out to be a problem.
2802 * This part of state machine should actually be handled by (i)pppd, but
2803 * that's too big of a change now. --kai
2804 */
2805
2806/* Actually, we might turn this into an advantage: deal with the RFC in
2807 * the old tradition of beeing generous on what we accept, but beeing
2808 * strict on what we send. Thus we should just
2809 * - accept compressed frames as soon as decompression is negotiated
2810 * - send compressed frames only when decomp *and* comp are negotiated
2811 * - drop rx compressed frames if we cannot decomp (instead of pushing them
2812 * up to ipppd)
2813 * and I tried to modify this file according to that. --abp
2814 */
2815
2816static void isdn_ppp_send_ccp(isdn_net_dev *net_dev, isdn_net_local *lp, struct sk_buff *skb)
2817{
2818 struct ippp_struct *mis,*is;
2819 int proto, slot = lp->ppp_slot;
2820 unsigned char *data;
2821
2822 if(!skb || skb->len < 3)
2823 return;
2824 if (slot < 0 || slot >= ISDN_MAX_CHANNELS) {
2825 printk(KERN_ERR "%s: lp->ppp_slot(%d) out of range\n",
2826 __FUNCTION__, slot);
2827 return;
2828 }
2829 is = ippp_table[slot];
2830 /* Daemon may send with or without address and control field comp */
2831 data = skb->data;
2832 if(!(is->pppcfg & SC_COMP_AC) && data[0] == 0xff && data[1] == 0x03) {
2833 data += 2;
2834 if(skb->len < 5)
2835 return;
2836 }
2837
2838 proto = ((int)data[0]<<8)+data[1];
2839 if(proto != PPP_CCP && proto != PPP_CCPFRAG)
2840 return;
2841
2842 printk(KERN_DEBUG "Received CCP frame from daemon:\n");
2843 isdn_ppp_frame_log("ccp-xmit", skb->data, skb->len, 32, is->unit,lp->ppp_slot);
2844
2845 if (lp->master) {
2846 slot = ((isdn_net_local *) (lp->master->priv))->ppp_slot;
Eric Sesterhenn052bb882006-04-11 17:29:17 -07002847 if (slot < 0 || slot >= ISDN_MAX_CHANNELS) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002848 printk(KERN_ERR "%s: slot(%d) out of range\n",
2849 __FUNCTION__, slot);
2850 return;
2851 }
2852 mis = ippp_table[slot];
2853 } else
2854 mis = is;
2855 if (mis != is)
2856 printk(KERN_DEBUG "isdn_ppp: Ouch! Master CCP sends on slave slot!\n");
2857
2858 switch(data[2]) {
2859 case CCP_CONFREQ:
2860 if(is->debug & 0x10)
2861 printk(KERN_DEBUG "Disable decompression here!\n");
2862 if(proto == PPP_CCP)
2863 is->compflags &= ~SC_DECOMP_ON;
2864 else
2865 is->compflags &= ~SC_LINK_DECOMP_ON;
2866 break;
2867 case CCP_TERMREQ:
2868 case CCP_TERMACK:
2869 if(is->debug & 0x10)
2870 printk(KERN_DEBUG "Disable (de)compression here!\n");
2871 if(proto == PPP_CCP)
2872 is->compflags &= ~(SC_DECOMP_ON|SC_COMP_ON);
2873 else
2874 is->compflags &= ~(SC_LINK_DECOMP_ON|SC_LINK_COMP_ON);
2875 break;
2876 case CCP_CONFACK:
2877 /* if we SEND an ackowledge we can/must enable the compressor */
2878 if(is->debug & 0x10)
2879 printk(KERN_DEBUG "Enable compression here!\n");
2880 if(proto == PPP_CCP) {
2881 if (!is->compressor)
2882 break;
2883 is->compflags |= SC_COMP_ON;
2884 } else {
2885 if (!is->compressor)
2886 break;
2887 is->compflags |= SC_LINK_COMP_ON;
2888 }
2889 break;
2890 case CCP_RESETACK:
2891 /* If we send a ACK we should reset our compressor */
2892 if(is->debug & 0x10)
2893 printk(KERN_DEBUG "Reset decompression state here!\n");
2894 printk(KERN_DEBUG "ResetAck from daemon passed by\n");
2895 if(proto == PPP_CCP) {
2896 /* link to master? */
2897 if(is->compressor && is->comp_stat)
2898 is->compressor->reset(is->comp_stat, 0, 0,
2899 NULL, 0, NULL);
2900 is->compflags &= ~SC_COMP_DISCARD;
2901 }
2902 else {
2903 if(is->link_compressor && is->link_comp_stat)
2904 is->link_compressor->reset(is->link_comp_stat,
2905 0, 0, NULL, 0, NULL);
2906 is->compflags &= ~SC_LINK_COMP_DISCARD;
2907 }
2908 break;
2909 case CCP_RESETREQ:
2910 /* Just let it pass by */
2911 printk(KERN_DEBUG "ResetReq from daemon passed by\n");
2912 break;
2913 }
2914}
2915
2916int isdn_ppp_register_compressor(struct isdn_ppp_compressor *ipc)
2917{
2918 ipc->next = ipc_head;
2919 ipc->prev = NULL;
2920 if(ipc_head) {
2921 ipc_head->prev = ipc;
2922 }
2923 ipc_head = ipc;
2924 return 0;
2925}
2926
2927int isdn_ppp_unregister_compressor(struct isdn_ppp_compressor *ipc)
2928{
2929 if(ipc->prev)
2930 ipc->prev->next = ipc->next;
2931 else
2932 ipc_head = ipc->next;
2933 if(ipc->next)
2934 ipc->next->prev = ipc->prev;
2935 ipc->prev = ipc->next = NULL;
2936 return 0;
2937}
2938
2939static int isdn_ppp_set_compressor(struct ippp_struct *is, struct isdn_ppp_comp_data *data)
2940{
2941 struct isdn_ppp_compressor *ipc = ipc_head;
2942 int ret;
2943 void *stat;
2944 int num = data->num;
2945
2946 if(is->debug & 0x10)
2947 printk(KERN_DEBUG "[%d] Set %s type %d\n",is->unit,
2948 (data->flags&IPPP_COMP_FLAG_XMIT)?"compressor":"decompressor",num);
2949
2950 /* If is has no valid reset state vector, we cannot allocate a
2951 decompressor. The decompressor would cause reset transactions
2952 sooner or later, and they need that vector. */
2953
2954 if(!(data->flags & IPPP_COMP_FLAG_XMIT) && !is->reset) {
2955 printk(KERN_ERR "ippp_ccp: no reset data structure - can't"
2956 " allow decompression.\n");
2957 return -ENOMEM;
2958 }
2959
2960 while(ipc) {
2961 if(ipc->num == num) {
2962 stat = ipc->alloc(data);
2963 if(stat) {
2964 ret = ipc->init(stat,data,is->unit,0);
2965 if(!ret) {
2966 printk(KERN_ERR "Can't init (de)compression!\n");
2967 ipc->free(stat);
2968 stat = NULL;
2969 break;
2970 }
2971 }
2972 else {
2973 printk(KERN_ERR "Can't alloc (de)compression!\n");
2974 break;
2975 }
2976
2977 if(data->flags & IPPP_COMP_FLAG_XMIT) {
2978 if(data->flags & IPPP_COMP_FLAG_LINK) {
2979 if(is->link_comp_stat)
2980 is->link_compressor->free(is->link_comp_stat);
2981 is->link_comp_stat = stat;
2982 is->link_compressor = ipc;
2983 }
2984 else {
2985 if(is->comp_stat)
2986 is->compressor->free(is->comp_stat);
2987 is->comp_stat = stat;
2988 is->compressor = ipc;
2989 }
2990 }
2991 else {
2992 if(data->flags & IPPP_COMP_FLAG_LINK) {
2993 if(is->link_decomp_stat)
2994 is->link_decompressor->free(is->link_decomp_stat);
2995 is->link_decomp_stat = stat;
2996 is->link_decompressor = ipc;
2997 }
2998 else {
2999 if(is->decomp_stat)
3000 is->decompressor->free(is->decomp_stat);
3001 is->decomp_stat = stat;
3002 is->decompressor = ipc;
3003 }
3004 }
3005 return 0;
3006 }
3007 ipc = ipc->next;
3008 }
3009 return -EINVAL;
3010}