Commit a0097af3 authored by Kai Germaschewski's avatar Kai Germaschewski

ISDN/PPP: dynamically allocate struct ipppd, further cleanups

o Now that all the infrastructure is in place, struct ipppd can
  easily be allocated when /dev/ipppX is opened and freed
  in the destructor.
o Separate the mix of state and flags in struct ipppd::state.
  We only have three states, open, assigned and connected, and
  additionally two flags to jump out of poll(), returning POLLHUP /
  POLLIN.
parent 1f4fa314
...@@ -47,7 +47,7 @@ ...@@ -47,7 +47,7 @@
#endif #endif
#define isdn_BUG() \ #define isdn_BUG() \
do { printk(KERN_WARNING "ISDN Bug at %s:%d\n", __FILE__, __LINE__); \ do { printk(KERN_WARNING "ISDN BUG at %s:%d\n", __FILE__, __LINE__); \
} while(0) } while(0)
#define HERE printk("%s:%d (%s)\n", __FILE__, __LINE__, __FUNCTION__) #define HERE printk("%s:%d (%s)\n", __FILE__, __LINE__, __FUNCTION__)
......
...@@ -21,8 +21,23 @@ ...@@ -21,8 +21,23 @@
#include "isdn_ppp_ccp.h" #include "isdn_ppp_ccp.h"
#include "isdn_net.h" #include "isdn_net.h"
/* ipppd::flags */
enum {
IPPPD_FL_HUP = 0x01,
IPPPD_FL_WAKEUP = 0x02,
};
/* ipppd::state */
enum {
IPPPD_ST_OPEN,
IPPPD_ST_ASSIGNED,
IPPPD_ST_CONNECTED,
};
struct ipppd { struct ipppd {
struct list_head ipppds;
int state; int state;
int flags;
struct sk_buff_head rq; struct sk_buff_head rq;
wait_queue_head_t wq; wait_queue_head_t wq;
struct isdn_net_dev_s *idev; struct isdn_net_dev_s *idev;
...@@ -32,6 +47,16 @@ struct ipppd { ...@@ -32,6 +47,16 @@ struct ipppd {
atomic_t refcnt; atomic_t refcnt;
}; };
#define IPPPD_DEBUG
#ifdef IPPPD_DEBUG
#define ipppd_debug(i, fmt, arg...) \
printk(KERN_DEBUG "ipppd %p state %#x %s:" fmt "\n", (i), (i)->state, \
__FUNCTION__ , ## arg)
#else
#define ipppd_debug(...) do { } while (0)
#endif
/* We use reference counting for struct ipppd. It is alloced on /* We use reference counting for struct ipppd. It is alloced on
* open() on /dev/ipppX and saved into file->private, making for one * open() on /dev/ipppX and saved into file->private, making for one
* reference. release() will release this reference, after all other * reference. release() will release this reference, after all other
...@@ -90,17 +115,16 @@ static int isdn_ppp_bundle(struct ipppd *, int unit); ...@@ -90,17 +115,16 @@ static int isdn_ppp_bundle(struct ipppd *, int unit);
char *isdn_ppp_revision = "$Revision: 1.85.6.9 $"; char *isdn_ppp_revision = "$Revision: 1.85.6.9 $";
#define NR_IPPPDS 64
static spinlock_t ipppds_lock = SPIN_LOCK_UNLOCKED; static spinlock_t ipppds_lock = SPIN_LOCK_UNLOCKED;
static struct ipppd *ipppds[NR_IPPPDS]; static LIST_HEAD(ipppds);
static void static void
ipppd_destroy(struct ipppd *ipppd) ipppd_destroy(struct ipppd *ipppd)
{ {
HERE; HERE;
ipppd->state = 0; skb_queue_purge(&ipppd->rq);
kfree(ipppd);
} }
static inline struct ipppd * static inline struct ipppd *
...@@ -111,26 +135,6 @@ __ipppd_get(struct ipppd *ipppd) ...@@ -111,26 +135,6 @@ __ipppd_get(struct ipppd *ipppd)
return ipppd; return ipppd;
} }
static inline struct ipppd *
ipppd_get(int slot)
{
unsigned long flags;
struct ipppd *ipppd;
if (slot < 0 || slot >= NR_IPPPDS)
return NULL;
spin_lock_irqsave(&ipppds_lock, flags);
ipppd = ipppds[slot];
if (!ipppd)
goto out;
__ipppd_get(ipppd);
out:
spin_unlock_irqrestore(&ipppds_lock, flags);
return ipppd;
}
static inline void static inline void
ipppd_put(struct ipppd *ipppd) ipppd_put(struct ipppd *ipppd)
{ {
...@@ -193,41 +197,28 @@ isdn_ppp_push_header(isdn_net_dev *idev, struct sk_buff *skb, u16 proto) ...@@ -193,41 +197,28 @@ isdn_ppp_push_header(isdn_net_dev *idev, struct sk_buff *skb, u16 proto)
static void static void
isdn_ppp_unbind(isdn_net_dev *idev) isdn_ppp_unbind(isdn_net_dev *idev)
{ {
struct ipppd *is; struct ipppd *is = idev->ipppd;
// FIXME much of this wants to rather happen when disconnected()
#ifdef CONFIG_ISDN_MPP
spin_lock(&idev->pb->lock);
if (lp->netdev->pb->ref_ct == 1) /* last link in queue? */
isdn_ppp_mp_cleanup(lp);
lp->netdev->pb->ref_ct--;
spin_unlock(&lp->netdev->pb->lock);
#endif /* CONFIG_ISDN_MPP */
is = idev->ipppd;
if (!is) { if (!is) {
isdn_BUG(); isdn_BUG();
return; return;
} }
if (is->state & IPPP_CONNECT) {
is->state = IPPP_CLOSEWAIT; ipppd_debug(is, "");
wake_up(&is->wq);
} else if (is->state & IPPP_ASSIGNED) { if (is->state != IPPPD_ST_ASSIGNED)
is->state = IPPP_OPEN; /* fallback to 'OPEN but not ASSIGNED' state */
} else {
isdn_BUG(); isdn_BUG();
}
if (is->debug & 0x1) is->state = IPPPD_ST_OPEN;
printk(KERN_DEBUG "isdn_ppp_unbind %p %p\n", idev, is);
/* is->idev will be invalid shortly */
ippp_ccp_free(idev->ccp);
is->idev = NULL; is->idev = NULL;
/* lose the reference we took on isdn_ppp_bind */ /* lose the reference we took on isdn_ppp_bind */
ipppd_put(is); ipppd_put(is);
idev->ipppd = NULL; idev->ipppd = NULL;
ippp_ccp_free(idev->ccp);
return; return;
} }
...@@ -237,10 +228,10 @@ isdn_ppp_unbind(isdn_net_dev *idev) ...@@ -237,10 +228,10 @@ isdn_ppp_unbind(isdn_net_dev *idev)
int int
isdn_ppp_bind(isdn_net_dev *idev) isdn_ppp_bind(isdn_net_dev *idev)
{ {
int i;
int unit = 0; int unit = 0;
unsigned long flags; unsigned long flags;
int retval = 0; int retval = 0;
struct ipppd *ipppd;
if (idev->ipppd) { if (idev->ipppd) {
isdn_BUG(); isdn_BUG();
...@@ -250,42 +241,42 @@ isdn_ppp_bind(isdn_net_dev *idev) ...@@ -250,42 +241,42 @@ isdn_ppp_bind(isdn_net_dev *idev)
spin_lock_irqsave(&ipppds_lock, flags); spin_lock_irqsave(&ipppds_lock, flags);
if (idev->pppbind < 0) { /* device bound to ippp device ? */ if (idev->pppbind < 0) { /* device bound to ippp device ? */
struct list_head *l; struct list_head *l;
char exclusive[NR_IPPPDS]; /* exclusive flags */ char exclusive[ISDN_MAX_CHANNELS]; /* exclusive flags */
memset(exclusive, 0, NR_IPPPDS); memset(exclusive, 0, ISDN_MAX_CHANNELS);
/* step through net devices to find exclusive minors */ /* step through net devices to find exclusive minors */
list_for_each(l, &isdn_net_devs) { list_for_each(l, &isdn_net_devs) {
isdn_net_dev *p = list_entry(l, isdn_net_dev, global_list); isdn_net_dev *p = list_entry(l, isdn_net_dev, global_list);
if (p->pppbind >= 0 && p->pppbind < NR_IPPPDS) if (p->pppbind >= 0 && p->pppbind < ISDN_MAX_CHANNELS)
exclusive[p->pppbind] = 1; exclusive[p->pppbind] = 1;
} }
/* /*
* search a free device / slot * search a free device / slot
*/ */
for (i = 0; i < NR_IPPPDS; i++) { list_for_each_entry(ipppd, &ipppds, ipppds) {
if (!ipppds[i]) if (!ipppd)
continue; continue;
if (ipppds[i]->state != IPPP_OPEN) if (ipppd->state != IPPPD_ST_OPEN)
continue; continue;
if (!exclusive[ipppds[i]->minor]) if (!exclusive[ipppd->minor])
break;
break; break;
goto found;
} }
} else { } else {
for (i = 0; i < NR_IPPPDS; i++) { list_for_each_entry(ipppd, &ipppds, ipppds) {
if (!ipppds[i]) if (!ipppd)
continue; continue;
if (ipppds[i]->state != IPPP_OPEN) if (ipppd->state != IPPPD_ST_OPEN)
continue; continue;
if (ipppds[i]->minor == idev->pppbind) if (ipppd->minor == idev->pppbind)
break; goto found;
} }
} }
if (i >= NR_IPPPDS) {
printk(KERN_INFO "isdn_ppp_bind: no ipppd\n"); printk(KERN_INFO "isdn_ppp_bind: no ipppd\n");
retval = -ESRCH; retval = -ESRCH;
goto err; goto err;
}
found:
unit = isdn_ppp_if_get_unit(idev->name); /* get unit number from interface name .. ugly! */ unit = isdn_ppp_if_get_unit(idev->name); /* get unit number from interface name .. ugly! */
if (unit < 0) { if (unit < 0) {
printk(KERN_INFO "isdn_ppp_bind: illegal interface name %s.\n", idev->name); printk(KERN_INFO "isdn_ppp_bind: illegal interface name %s.\n", idev->name);
...@@ -293,11 +284,11 @@ isdn_ppp_bind(isdn_net_dev *idev) ...@@ -293,11 +284,11 @@ isdn_ppp_bind(isdn_net_dev *idev)
goto err; goto err;
} }
ipppds[i]->unit = unit; ipppd->unit = unit;
ipppds[i]->state = IPPP_OPEN | IPPP_ASSIGNED; ipppd->state = IPPPD_ST_ASSIGNED;
ipppds[i]->idev = idev; ipppd->idev = idev;
/* we hold a reference until isdn_ppp_unbind() */ /* we hold a reference until isdn_ppp_unbind() */
idev->ipppd = __ipppd_get(ipppds[i]); idev->ipppd = __ipppd_get(ipppd);
spin_unlock_irqrestore(&ipppds_lock, flags); spin_unlock_irqrestore(&ipppds_lock, flags);
idev->pppcfg = 0; /* config flags */ idev->pppcfg = 0; /* config flags */
...@@ -321,7 +312,7 @@ isdn_ppp_bind(isdn_net_dev *idev) ...@@ -321,7 +312,7 @@ isdn_ppp_bind(isdn_net_dev *idev)
#endif /* CONFIG_ISDN_MPP */ #endif /* CONFIG_ISDN_MPP */
out: out:
if (retval) { if (retval) {
idev->ipppd->state = IPPP_OPEN; idev->ipppd->state = IPPPD_ST_OPEN;
ipppd_put(idev->ipppd); ipppd_put(idev->ipppd);
idev->ipppd = NULL; idev->ipppd = NULL;
} }
...@@ -343,29 +334,39 @@ isdn_ppp_connected(isdn_net_dev *idev) ...@@ -343,29 +334,39 @@ isdn_ppp_connected(isdn_net_dev *idev)
{ {
struct ipppd *ipppd = idev->ipppd; struct ipppd *ipppd = idev->ipppd;
ipppd->state = IPPP_OPEN | IPPP_CONNECT | IPPP_NOBLOCK; ipppd_debug(ipppd, "");
ipppd->state = IPPPD_ST_CONNECTED;
ipppd->flags |= IPPPD_FL_WAKEUP;
wake_up(&ipppd->wq); wake_up(&ipppd->wq);
} }
/* static void
* isdn_ppp_get_slot isdn_ppp_disconnected(isdn_net_dev *idev)
*/
static int
isdn_ppp_get_slot(void)
{ {
int i; struct ipppd *ipppd = idev->ipppd;
unsigned long flags;
ipppd_debug(ipppd, "");
if (idev->pppcfg & SC_ENABLE_IP)
isdn_net_offline(idev);
if (ipppd->state != IPPPD_ST_CONNECTED)
isdn_BUG();
ipppd->state = IPPPD_ST_ASSIGNED;
ipppd->flags |= IPPPD_FL_HUP;
wake_up(&ipppd->wq);
#ifdef CONFIG_ISDN_MPP
spin_lock(&idev->pb->lock);
if (lp->netdev->pb->ref_ct == 1) /* last link in queue? */
isdn_ppp_mp_cleanup(lp);
lp->netdev->pb->ref_ct--;
spin_unlock(&lp->netdev->pb->lock);
#endif /* CONFIG_ISDN_MPP */
spin_lock_irqsave(&ipppds_lock, flags);
for (i = 0; i < NR_IPPPDS; i++) {
if (ipppds[i]->state == 0) {
ipppds[i]->state = IPPP_OPEN;
break;
}
}
spin_unlock_irqrestore(&ipppds_lock, flags);
return (i < NR_IPPPDS) ? i : -1;
} }
/* /*
...@@ -375,24 +376,31 @@ isdn_ppp_get_slot(void) ...@@ -375,24 +376,31 @@ isdn_ppp_get_slot(void)
static int static int
ipppd_open(struct inode *ino, struct file *file) ipppd_open(struct inode *ino, struct file *file)
{ {
uint minor = minor(ino->i_rdev) - ISDN_MINOR_PPP; unsigned long flags;
int slot; unsigned int minor = minor(ino->i_rdev) - ISDN_MINOR_PPP;
struct ipppd *is; struct ipppd *ipppd;
slot = isdn_ppp_get_slot(); ipppd = kmalloc(sizeof(*ipppd), GFP_KERNEL);
if (slot < 0) if (!ipppd)
return -EBUSY; return -ENOMEM;
is = ipppds[slot]; memset(ipppd, 0, sizeof(*ipppd));
file->private_data = is; atomic_set(&ipppd->refcnt, 0);
__ipppd_get(is);
printk(KERN_DEBUG "ippp, open, slot: %d, minor: %d, state: %04x\n", slot, minor, is->state); /* file->private_data holds a reference */
file->private_data = __ipppd_get(ipppd);
is->idev = NULL; ipppd->unit = -1; /* set by isdn_ppp_bind */
is->unit = -1; /* set, when we have our interface */ ipppd->minor = minor;
init_waitqueue_head(&is->wq); ipppd->state = IPPPD_ST_OPEN;
is->minor = minor; init_waitqueue_head(&ipppd->wq);
skb_queue_head_init(&ipppd->rq);
spin_lock_irqsave(&ipppds, flags);
list_add(&ipppd->ipppds, &ipppds);
spin_unlock_irqrestore(&ipppds, flags);
ipppd_debug(ipppd, "minor %d", minor);
return 0; return 0;
} }
...@@ -403,30 +411,20 @@ ipppd_open(struct inode *ino, struct file *file) ...@@ -403,30 +411,20 @@ ipppd_open(struct inode *ino, struct file *file)
static int static int
ipppd_release(struct inode *ino, struct file *file) ipppd_release(struct inode *ino, struct file *file)
{ {
uint minor = minor(ino->i_rdev) - ISDN_MINOR_PPP; unsigned long flags;
struct ipppd *is; struct ipppd *ipppd = file->private_data;
lock_kernel();
is = file->private_data; ipppd_debug(ipppd, "");
if (is->debug & 0x1) if (ipppd->state == IPPPD_ST_CONNECTED)
printk(KERN_DEBUG "ippp: release, minor: %d %p\n", minor, is->idev); isdn_net_hangup(ipppd->idev);
if (is->idev) { /* a lp address says: this link is still up */ spin_lock_irqsave(&ipppds, flags);
/* list_del(&ipppd->ipppds);
* isdn_net_hangup() calls isdn_ppp_free() spin_unlock_irqrestore(&ipppds, flags);
* isdn_ppp_free() sets is->lp to NULL and lp->ppp_slot to -1
* removing the IPPP_CONNECT flag omits calling of isdn_ppp_wakeup_daemon()
*/
is->state &= ~IPPP_CONNECT;
isdn_net_hangup(is->idev);
}
skb_queue_purge(&is->rq);
ipppd_put(is); ipppd_put(ipppd);
unlock_kernel();
return 0; return 0;
} }
...@@ -473,9 +471,6 @@ ipppd_ioctl(struct inode *ino, struct file *file, unsigned int cmd, unsigned lon ...@@ -473,9 +471,6 @@ ipppd_ioctl(struct inode *ino, struct file *file, unsigned int cmd, unsigned lon
if (is->debug & 0x1) if (is->debug & 0x1)
printk(KERN_DEBUG "isdn_ppp_ioctl: minor: %d cmd: %x state: %x\n", is->minor, cmd, is->state); printk(KERN_DEBUG "isdn_ppp_ioctl: minor: %d cmd: %x state: %x\n", is->minor, cmd, is->state);
if (!(is->state & IPPP_OPEN))
return -EINVAL;
switch (cmd) { switch (cmd) {
case PPPIOCBUNDLE: case PPPIOCBUNDLE:
#ifdef CONFIG_ISDN_MPP #ifdef CONFIG_ISDN_MPP
...@@ -642,30 +637,23 @@ ipppd_poll(struct file *file, poll_table * wait) ...@@ -642,30 +637,23 @@ ipppd_poll(struct file *file, poll_table * wait)
is = file->private_data; is = file->private_data;
if (is->debug & 0x2) ipppd_debug(is, "");
printk(KERN_DEBUG "isdn_ppp_poll: minor: %d\n",
minor(file->f_dentry->d_inode->i_rdev));
/* just registers wait_queue hook. This doesn't really wait. */ /* just registers wait_queue hook. This doesn't really wait. */
poll_wait(file, &is->wq, wait); poll_wait(file, &is->wq, wait);
if (!(is->state & IPPP_OPEN)) { if (is->flags & IPPPD_FL_HUP) {
if(is->state == IPPP_CLOSEWAIT) {
mask = POLLHUP; mask = POLLHUP;
goto out; goto out;
} }
printk(KERN_DEBUG "isdn_ppp: device not open\n");
mask = POLLERR;
goto out;
}
/* we're always ready to send .. */ /* we're always ready to send .. */
mask = POLLOUT | POLLWRNORM; mask = POLLOUT | POLLWRNORM;
/* /*
* if IPPP_NOBLOCK is set we return even if we have nothing to read * if IPPP_FL_WAKEUP is set we return even if we have nothing to read
*/ */
if (!skb_queue_empty(&is->rq) || is->state & IPPP_NOBLOCK) { if (!skb_queue_empty(&is->rq) || is->flags & IPPPD_FL_WAKEUP) {
is->state &= ~IPPP_NOBLOCK; is->flags &= ~IPPPD_FL_WAKEUP;
mask |= POLLIN | POLLRDNORM; mask |= POLLIN | POLLRDNORM;
set_current_state(TASK_INTERRUPTIBLE); // FIXME set_current_state(TASK_INTERRUPTIBLE); // FIXME
schedule_timeout(HZ); schedule_timeout(HZ);
...@@ -686,8 +674,8 @@ ipppd_queue_read(struct ipppd *is, u16 proto, unsigned char *buf, int len) ...@@ -686,8 +674,8 @@ ipppd_queue_read(struct ipppd *is, u16 proto, unsigned char *buf, int len)
unsigned char *p; unsigned char *p;
int retval; int retval;
if (!(is->state & IPPP_CONNECT)) { if (is->state != IPPPD_ST_CONNECTED) {
printk(KERN_DEBUG "ippp: device not activated.\n"); printk(KERN_DEBUG "ippp: device not connected.\n");
retval = -ENOTCONN; retval = -ENOTCONN;
goto out; goto out;
} }
...@@ -733,10 +721,6 @@ ipppd_read(struct file *file, char *buf, size_t count, loff_t *off) ...@@ -733,10 +721,6 @@ ipppd_read(struct file *file, char *buf, size_t count, loff_t *off)
is = file->private_data; is = file->private_data;
if (!(is->state & IPPP_OPEN)) {
retval = 0;
goto out;
}
skb = skb_dequeue(&is->rq); skb = skb_dequeue(&is->rq);
if (!skb) { if (!skb) {
retval = -EAGAIN; retval = -EAGAIN;
...@@ -778,8 +762,11 @@ ipppd_write(struct file *file, const char *buf, size_t count, loff_t *off) ...@@ -778,8 +762,11 @@ ipppd_write(struct file *file, const char *buf, size_t count, loff_t *off)
is = file->private_data; is = file->private_data;
if (!(is->state & IPPP_CONNECT)) { ipppd_debug(is, "");
retval = 0;
if (is->state != IPPPD_ST_CONNECTED) {
HERE;
retval = -ENOTCONN;
goto out; goto out;
} }
...@@ -863,36 +850,17 @@ struct file_operations isdn_ppp_fops = ...@@ -863,36 +850,17 @@ struct file_operations isdn_ppp_fops =
int int
isdn_ppp_init(void) isdn_ppp_init(void)
{ {
int i;
#ifdef CONFIG_ISDN_MPP #ifdef CONFIG_ISDN_MPP
if( isdn_ppp_mp_bundle_array_init() < 0 ) if( isdn_ppp_mp_bundle_array_init() < 0 )
return -ENOMEM; return -ENOMEM;
#endif /* CONFIG_ISDN_MPP */ #endif /* CONFIG_ISDN_MPP */
for (i = 0; i < NR_IPPPDS; i++) {
ipppds[i] = kmalloc(sizeof(struct ipppd), GFP_KERNEL);
if (!ipppds[i]) {
printk(KERN_WARNING "isdn_ppp_init: Could not alloc ippp_table\n");
for (i--; i >= 0; i++)
kfree(ipppds[i]);
return -ENOMEM;
}
memset(ipppds[i], 0, sizeof(struct ipppd));
ipppds[i]->state = 0;
skb_queue_head_init(&ipppds[i]->rq);
}
return 0; return 0;
} }
void void
isdn_ppp_cleanup(void) isdn_ppp_cleanup(void)
{ {
int i;
for (i = 0; i < NR_IPPPDS; i++)
kfree(ipppds[i]);
#ifdef CONFIG_ISDN_MPP #ifdef CONFIG_ISDN_MPP
if (isdn_ppp_bundle_arr) if (isdn_ppp_bundle_arr)
kfree(isdn_ppp_bundle_arr); kfree(isdn_ppp_bundle_arr);
...@@ -1765,11 +1733,11 @@ isdn_ppp_bundle(struct ipppd *is, int unit) ...@@ -1765,11 +1733,11 @@ isdn_ppp_bundle(struct ipppd *is, int unit)
*/ */
static int static int
isdn_ppp_dev_ioctl_stats(int slot, struct ifreq *ifr, struct net_device *dev) isdn_ppp_dev_ioctl_stats(struct ifreq *ifr, struct net_device *dev)
{ {
struct ppp_stats *res, t; struct ppp_stats *res, t;
struct ipppd *is;
isdn_net_local *lp = (isdn_net_local *) dev->priv; isdn_net_local *lp = (isdn_net_local *) dev->priv;
struct slcompress *slcomp;
int err; int err;
res = (struct ppp_stats *) ifr->ifr_ifru.ifru_data; res = (struct ppp_stats *) ifr->ifr_ifru.ifru_data;
...@@ -1789,9 +1757,7 @@ isdn_ppp_dev_ioctl_stats(int slot, struct ifreq *ifr, struct net_device *dev) ...@@ -1789,9 +1757,7 @@ isdn_ppp_dev_ioctl_stats(int slot, struct ifreq *ifr, struct net_device *dev)
t.p.ppp_obytes = lp->stats.tx_bytes; t.p.ppp_obytes = lp->stats.tx_bytes;
t.p.ppp_oerrors = lp->stats.tx_errors; t.p.ppp_oerrors = lp->stats.tx_errors;
#ifdef CONFIG_ISDN_PPP_VJ #ifdef CONFIG_ISDN_PPP_VJ
is = ipppd_get(slot); slcomp = lp->slcomp;
if (is) {
struct slcompress *slcomp = lp->slcomp;
if (slcomp) { if (slcomp) {
t.vj.vjs_packets = slcomp->sls_o_compressed + slcomp->sls_o_uncompressed; t.vj.vjs_packets = slcomp->sls_o_compressed + slcomp->sls_o_uncompressed;
t.vj.vjs_compressed = slcomp->sls_o_compressed; t.vj.vjs_compressed = slcomp->sls_o_compressed;
...@@ -1802,8 +1768,6 @@ isdn_ppp_dev_ioctl_stats(int slot, struct ifreq *ifr, struct net_device *dev) ...@@ -1802,8 +1768,6 @@ isdn_ppp_dev_ioctl_stats(int slot, struct ifreq *ifr, struct net_device *dev)
t.vj.vjs_uncompressedin = slcomp->sls_i_uncompressed; t.vj.vjs_uncompressedin = slcomp->sls_i_uncompressed;
t.vj.vjs_compressedin = slcomp->sls_i_compressed; t.vj.vjs_compressedin = slcomp->sls_i_compressed;
} }
ipppd_put(is);
}
#endif #endif
} }
if( copy_to_user(res, &t, sizeof(struct ppp_stats))) return -EFAULT; if( copy_to_user(res, &t, sizeof(struct ppp_stats))) return -EFAULT;
...@@ -1829,7 +1793,7 @@ isdn_ppp_dev_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd) ...@@ -1829,7 +1793,7 @@ isdn_ppp_dev_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd)
if(copy_to_user(r, PPP_VERSION, len)) error = -EFAULT; if(copy_to_user(r, PPP_VERSION, len)) error = -EFAULT;
break; break;
case SIOCGPPPSTATS: case SIOCGPPPSTATS:
error = isdn_ppp_dev_ioctl_stats(0, ifr, dev); error = isdn_ppp_dev_ioctl_stats(ifr, dev);
break; break;
default: default:
error = -EINVAL; error = -EINVAL;
...@@ -2132,13 +2096,6 @@ isdn_ppp_close(isdn_net_local *lp) ...@@ -2132,13 +2096,6 @@ isdn_ppp_close(isdn_net_local *lp)
ippp_ccp_free(lp->ccp); ippp_ccp_free(lp->ccp);
} }
static void
isdn_ppp_disconnected(isdn_net_dev *idev)
{
if (idev->pppcfg & SC_ENABLE_IP)
isdn_net_offline(idev);
}
struct isdn_netif_ops isdn_ppp_ops = { struct isdn_netif_ops isdn_ppp_ops = {
.hard_start_xmit = isdn_ppp_start_xmit, .hard_start_xmit = isdn_ppp_start_xmit,
.do_ioctl = isdn_ppp_dev_ioctl, .do_ioctl = isdn_ppp_dev_ioctl,
......
...@@ -26,12 +26,6 @@ isdn_ppp_frame_log(char *info, char *data, int len, int maxlen, ...@@ -26,12 +26,6 @@ isdn_ppp_frame_log(char *info, char *data, int len, int maxlen,
int int
isdn_ppp_strip_proto(struct sk_buff *skb); isdn_ppp_strip_proto(struct sk_buff *skb);
#define IPPP_OPEN 0x01
#define IPPP_CONNECT 0x02
#define IPPP_CLOSEWAIT 0x04
#define IPPP_NOBLOCK 0x08
#define IPPP_ASSIGNED 0x10
#define IPPP_MAX_HEADER 10 #define IPPP_MAX_HEADER 10
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment