Commit 713f2def authored by Kai Germaschewski's avatar Kai Germaschewski

ISDN/PPP: Move rest of CCP reset handling into isdn_ppp_ccp.c

Basically, all what was missing was a common way to xmit frames
given the isdn_net_dev / isdn_net_local. We have this now by means
of function pointers (still room for improvement there, though), so
now let's do things from isdn_ppp_ccp.c.
parent 949d4e94
...@@ -43,17 +43,28 @@ static void ...@@ -43,17 +43,28 @@ static void
isdn_ppp_receive_ccp(isdn_net_dev * net_dev, isdn_net_local * lp, isdn_ppp_receive_ccp(isdn_net_dev * net_dev, isdn_net_local * lp,
struct sk_buff *skb,int proto); struct sk_buff *skb,int proto);
static struct sk_buff *
isdn_ppp_dev_alloc_skb(void *priv, int len, int gfp_mask);
static void
isdn_ppp_dev_push_header(void *priv, struct sk_buff *skb, u16 proto);
static void
isdn_ppp_dev_xmit(void *priv, struct sk_buff *skb);
static struct sk_buff *
isdn_ppp_lp_alloc_skb(void *priv, int len, int gfp_mask);
static void
isdn_ppp_lp_push_header(void *priv, struct sk_buff *skb, u16 proto);
static void static void
isdn_ppp_send_ccp(isdn_net_dev *net_dev, isdn_net_local *lp, isdn_ppp_send_ccp(isdn_net_dev *net_dev, isdn_net_local *lp,
struct sk_buff *skb); struct sk_buff *skb);
/* New CCP stuff */ /* New CCP stuff */
static void static void
isdn_ppp_ccp_kick_up(void *priv); isdn_ppp_dev_kick_up(void *priv);
static void
isdn_ppp_ccp_xmit_reset(void *priv, int proto, unsigned char code,
unsigned char id, unsigned char *data, int len);
#ifdef CONFIG_ISDN_MPP #ifdef CONFIG_ISDN_MPP
static ippp_bundle * isdn_ppp_bundle_arr = NULL; static ippp_bundle * isdn_ppp_bundle_arr = NULL;
...@@ -107,7 +118,7 @@ isdn_ppp_frame_log(char *info, char *data, int len, int maxlen,int unit,int slot ...@@ -107,7 +118,7 @@ isdn_ppp_frame_log(char *info, char *data, int len, int maxlen,int unit,int slot
} }
void static void
isdn_ppp_push_header(isdn_net_dev *idev, struct sk_buff *skb, u16 proto) isdn_ppp_push_header(isdn_net_dev *idev, struct sk_buff *skb, u16 proto)
{ {
unsigned char *p; unsigned char *p;
...@@ -259,12 +270,17 @@ isdn_ppp_bind(isdn_net_dev *idev) ...@@ -259,12 +270,17 @@ isdn_ppp_bind(isdn_net_dev *idev)
/* seq no last seen, maybe set to bundle min, when joining? */ /* seq no last seen, maybe set to bundle min, when joining? */
idev->pppseq = -1; idev->pppseq = -1;
idev->ccp = ippp_ccp_alloc(PPP_COMPFRAG, idev, isdn_ppp_ccp_xmit_reset, idev->ccp = ippp_ccp_alloc();
isdn_ppp_ccp_kick_up);
if (!idev->ccp) { if (!idev->ccp) {
retval = -ENOMEM; retval = -ENOMEM;
goto out; goto out;
} }
idev->ccp->proto = PPP_COMPFRAG;
idev->ccp->priv = idev;
idev->ccp->alloc_skb = isdn_ppp_dev_alloc_skb;
idev->ccp->push_header = isdn_ppp_dev_push_header;
idev->ccp->xmit = isdn_ppp_dev_xmit;
idev->ccp->kick_up = isdn_ppp_dev_kick_up;
#ifdef CONFIG_ISDN_MPP #ifdef CONFIG_ISDN_MPP
retval = isdn_ppp_mp_init(lp, NULL); retval = isdn_ppp_mp_init(lp, NULL);
...@@ -1930,14 +1946,14 @@ isdn_ppp_hangup_slave(char *name) ...@@ -1930,14 +1946,14 @@ isdn_ppp_hangup_slave(char *name)
/* Push an empty CCP Data Frame up to the daemon to wake it up and let it /* Push an empty CCP Data Frame up to the daemon to wake it up and let it
generate a CCP Reset-Request or tear down CCP altogether */ generate a CCP Reset-Request or tear down CCP altogether */
static void isdn_ppp_ccp_kick_up(void *priv) static void isdn_ppp_dev_kick_up(void *priv)
{ {
isdn_net_dev *idev = priv; isdn_net_dev *idev = priv;
isdn_ppp_fill_rq(NULL, 0, PPP_COMP, idev->ppp_slot); isdn_ppp_fill_rq(NULL, 0, PPP_COMP, idev->ppp_slot);
} }
static void isdn_ppp_ccp_lp_kick_up(void *priv) static void isdn_ppp_lp_kick_up(void *priv)
{ {
isdn_net_local *lp = priv; isdn_net_local *lp = priv;
isdn_net_dev *idev; isdn_net_dev *idev;
...@@ -1952,43 +1968,74 @@ static void isdn_ppp_ccp_lp_kick_up(void *priv) ...@@ -1952,43 +1968,74 @@ static void isdn_ppp_ccp_lp_kick_up(void *priv)
/* Send a CCP Reset-Request or Reset-Ack directly from the kernel. */ /* Send a CCP Reset-Request or Reset-Ack directly from the kernel. */
static void static struct sk_buff *
isdn_ppp_ccp_xmit_reset(void *priv, int proto, unsigned char code, __isdn_ppp_alloc_skb(isdn_net_dev *idev, int len, unsigned int gfp_mask)
unsigned char id, unsigned char *data, int len)
{ {
isdn_net_dev *idev = priv; int hl = IPPP_MAX_HEADER + isdn_slot_hdrlen(idev->isdn_slot);
struct sk_buff *skb; struct sk_buff *skb;
unsigned char *p;
int hl;
hl = isdn_slot_hdrlen(idev->isdn_slot); skb = alloc_skb(hl + len, gfp_mask);
skb = alloc_skb(len + hl + IPPP_MAX_HEADER, GFP_ATOMIC); if (!skb)
if(!skb) { return NULL;
printk(KERN_WARNING
"ippp: CCP cannot send reset - out of memory\n"); skb_reserve(skb, hl);
return; return skb;
}
static struct sk_buff *
isdn_ppp_dev_alloc_skb(void *priv, int len, int gfp_mask)
{
isdn_net_dev *idev = priv;
return __isdn_ppp_alloc_skb(idev, len, gfp_mask);
}
static struct sk_buff *
isdn_ppp_lp_alloc_skb(void *priv, int len, int gfp_mask)
{
isdn_net_local *lp = priv;
isdn_net_dev *idev;
if (list_empty(&lp->online)) {
isdn_BUG();
return NULL;
} }
skb_reserve(skb, hl+16); idev = list_entry(lp->online.next, isdn_net_dev, online);
return __isdn_ppp_alloc_skb(idev, len, gfp_mask);
}
static void
isdn_ppp_dev_push_header(void *priv, struct sk_buff *skb, u16 proto)
{
isdn_net_dev *idev = priv;
isdn_ppp_push_header(idev, skb, proto); isdn_ppp_push_header(idev, skb, proto);
}
p = skb_put(skb, 4); static void
p += put_u8 (p, code); isdn_ppp_lp_push_header(void *priv, struct sk_buff *skb, u16 proto)
p += put_u8 (p, id); {
p += put_u16(p, len + 4); isdn_net_local *lp = priv;
isdn_net_dev *idev;
if (len) if (list_empty(&lp->online)) {
memcpy(skb_put(skb, len), data, len); isdn_BUG();
return;
}
idev = list_entry(lp->online.next, isdn_net_dev, online);
isdn_ppp_push_header(idev, skb, proto);
}
isdn_ppp_frame_log("ccp-xmit", skb->data, skb->len, 32, -1, static void
idev->ppp_slot); isdn_ppp_dev_xmit(void *priv, struct sk_buff *skb)
{
isdn_net_dev *idev = priv;
isdn_net_write_super(idev, skb); isdn_net_write_super(idev, skb);
} }
static void isdn_ppp_ccp_lp_xmit_reset(void *priv, int proto, static void
unsigned char code, unsigned char id, isdn_ppp_lp_xmit(void *priv, struct sk_buff *skb)
unsigned char *data, int len)
{ {
isdn_net_local *lp = priv; isdn_net_local *lp = priv;
isdn_net_dev *idev; isdn_net_dev *idev;
...@@ -1998,7 +2045,7 @@ static void isdn_ppp_ccp_lp_xmit_reset(void *priv, int proto, ...@@ -1998,7 +2045,7 @@ static void isdn_ppp_ccp_lp_xmit_reset(void *priv, int proto,
return; return;
} }
idev = list_entry(lp->online.next, isdn_net_dev, online); idev = list_entry(lp->online.next, isdn_net_dev, online);
isdn_ppp_ccp_xmit_reset(idev, proto, code, id, data, len); isdn_net_write_super(idev, skb);
} }
...@@ -2087,10 +2134,15 @@ isdn_ppp_open(isdn_net_local *lp) ...@@ -2087,10 +2134,15 @@ isdn_ppp_open(isdn_net_local *lp)
#ifdef CONFIG_ISDN_PPP_VJ #ifdef CONFIG_ISDN_PPP_VJ
lp->slcomp = slhc_init(16, 16); lp->slcomp = slhc_init(16, 16);
#endif #endif
lp->ccp = ippp_ccp_alloc(PPP_COMPFRAG, lp, isdn_ppp_ccp_lp_xmit_reset, lp->ccp = ippp_ccp_alloc();
isdn_ppp_ccp_lp_kick_up);
if (!lp->ccp) if (!lp->ccp)
return -ENOMEM; return -ENOMEM;
lp->ccp->proto = PPP_COMP;
lp->ccp->priv = lp;
lp->ccp->alloc_skb = isdn_ppp_lp_alloc_skb;
lp->ccp->push_header = isdn_ppp_lp_push_header;
lp->ccp->xmit = isdn_ppp_lp_xmit;
lp->ccp->kick_up = isdn_ppp_lp_kick_up;
return 0; return 0;
} }
......
#include "isdn_ppp_ccp.h" #include "isdn_ppp_ccp.h"
#include "isdn_common.h" #include "isdn_common.h"
#include "isdn_net.h"
#include "isdn_ppp.h" #include "isdn_ppp.h"
#include <linux/ppp-comp.h> #include <linux/ppp-comp.h>
...@@ -52,13 +53,28 @@ ippp_ccp_reset_free_state(struct ippp_ccp *ccp, unsigned char id) ...@@ -52,13 +53,28 @@ ippp_ccp_reset_free_state(struct ippp_ccp *ccp, unsigned char id)
ccp->reset->rs[id] = NULL; ccp->reset->rs[id] = NULL;
} }
static inline void static void
do_xmit_reset(struct ippp_ccp *ccp, unsigned char code, unsigned char id, do_xmit_reset(struct ippp_ccp *ccp, unsigned char code, unsigned char id,
unsigned char *data, int len) unsigned char *data, int len)
{ {
ccp->xmit_reset(ccp->priv, struct sk_buff *skb;
ccp->proto == PPP_COMP ? PPP_CCP : PPP_CCPFRAG, unsigned char *p;
code, id, data, len); u16 proto = ccp->proto == PPP_COMP ? PPP_CCP : PPP_CCPFRAG;
skb = ccp->alloc_skb(ccp->priv, 4 + len, GFP_ATOMIC);
ccp->push_header(ccp->priv, skb, proto);
p = skb_put(skb, 4);
p += put_u8 (p, code);
p += put_u8 (p, id);
p += put_u16(p, len + 4);
if (len)
memcpy(skb_put(skb, len), data, len);
isdn_ppp_frame_log("ccp-xmit", skb->data, skb->len, 32, -1, -1);
ccp->xmit(ccp->priv, skb);
} }
/* The timer callback function which is called when a ResetReq has timed out, /* The timer callback function which is called when a ResetReq has timed out,
...@@ -181,17 +197,12 @@ ippp_ccp_reset_xmit(struct ippp_ccp *ccp, ...@@ -181,17 +197,12 @@ ippp_ccp_reset_xmit(struct ippp_ccp *ccp,
/* ====================================================================== */ /* ====================================================================== */
struct ippp_ccp * struct ippp_ccp *
ippp_ccp_alloc(int proto, void *priv, ippp_ccp_alloc(void)
void (*xmit_reset)(void *priv, int proto, unsigned char code,
unsigned char id, unsigned char *data,
int len),
void (*kick_up)(void *priv))
{ {
struct ippp_ccp *ccp; struct ippp_ccp *ccp;
ccp = kmalloc(sizeof(*ccp), GFP_ATOMIC); // FIXME ccp = kmalloc(sizeof(*ccp), GFP_ATOMIC); // FIXME
memset(ccp, 0, sizeof(*ccp)); memset(ccp, 0, sizeof(*ccp));
ccp->proto = proto;
ccp->mru = 1524; /* MRU, default 1524 */ ccp->mru = 1524; /* MRU, default 1524 */
ccp->reset = kmalloc(sizeof(*ccp->reset), GFP_ATOMIC); // FIXME alloc together? ccp->reset = kmalloc(sizeof(*ccp->reset), GFP_ATOMIC); // FIXME alloc together?
if (!ccp->reset) { if (!ccp->reset) {
...@@ -199,9 +210,6 @@ ippp_ccp_alloc(int proto, void *priv, ...@@ -199,9 +210,6 @@ ippp_ccp_alloc(int proto, void *priv,
return NULL; return NULL;
} }
memset(ccp->reset, 0, sizeof(*ccp->reset)); memset(ccp->reset, 0, sizeof(*ccp->reset));
ccp->priv = priv;
ccp->xmit_reset = xmit_reset;
ccp->kick_up = kick_up;
return ccp; return ccp;
} }
...@@ -231,9 +239,9 @@ ippp_ccp_set_mru(struct ippp_ccp *ccp, unsigned int mru) ...@@ -231,9 +239,9 @@ ippp_ccp_set_mru(struct ippp_ccp *ccp, unsigned int mru)
} }
unsigned int unsigned int
ippp_ccp_get_flags(struct ippp_ccp *ccp); ippp_ccp_get_flags(struct ippp_ccp *ccp)
{ {
return idev->ccp->compflags & (SC_DC_ERROR|SC_DC_FERROR); return ccp->compflags & (SC_DC_ERROR|SC_DC_FERROR);
} }
/* /*
......
...@@ -26,17 +26,14 @@ struct ippp_ccp { ...@@ -26,17 +26,14 @@ struct ippp_ccp {
int mru; int mru;
int debug; int debug;
void *priv; void *priv;
void (*xmit_reset)(void *priv, int proto, unsigned char code, void (*xmit)(void *priv, struct sk_buff *skb);
unsigned char id, unsigned char *data, int len);
void (*kick_up)(void *priv); void (*kick_up)(void *priv);
void (*push_header)(void *priv, struct sk_buff *skb, u16);
struct sk_buff *(*alloc_skb)(void *priv, int len, int gfp_mask);
}; };
struct ippp_ccp * struct ippp_ccp *
ippp_ccp_alloc(int proto, void *priv, ippp_ccp_alloc(void);
void (*xmit_reset)(void *priv, int proto, unsigned char code,
unsigned char id, unsigned char *data,
int len),
void (*kick_up)(void *priv));
void void
ippp_ccp_free(struct ippp_ccp *ccp); ippp_ccp_free(struct ippp_ccp *ccp);
......
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