|  | /* | 
|  | * CAPI encoder/decoder for | 
|  | * Portugal Telecom CAPI 2.0 | 
|  | * | 
|  | * Copyright (C) 1996 Universidade de Lisboa | 
|  | * | 
|  | * Written by Pedro Roque Marques (roque@di.fc.ul.pt) | 
|  | * | 
|  | * This software may be used and distributed according to the terms of | 
|  | * the GNU General Public License, incorporated herein by reference. | 
|  | * | 
|  | * Not compatible with the AVM Gmbh. CAPI 2.0 | 
|  | * | 
|  | */ | 
|  |  | 
|  | /* | 
|  | *        Documentation: | 
|  | *        - "Common ISDN API - Perfil Português - Versão 2.1", | 
|  | *           Telecom Portugal, Fev 1992. | 
|  | *        - "Common ISDN API - Especificação de protocolos para | 
|  | *           acesso aos canais B", Inesc, Jan 1994. | 
|  | */ | 
|  |  | 
|  | /* | 
|  | *        TODO: better decoding of Information Elements | 
|  | *              for debug purposes mainly | 
|  | *              encode our number in CallerPN and ConnectedPN | 
|  | */ | 
|  |  | 
|  | #include <linux/string.h> | 
|  | #include <linux/kernel.h> | 
|  |  | 
|  | #include <linux/types.h> | 
|  | #include <linux/slab.h> | 
|  | #include <linux/mm.h> | 
|  |  | 
|  | #include <linux/skbuff.h> | 
|  |  | 
|  | #include <asm/io.h> | 
|  | #include <asm/string.h> | 
|  |  | 
|  | #include <linux/isdnif.h> | 
|  |  | 
|  | #include "pcbit.h" | 
|  | #include "edss1.h" | 
|  | #include "capi.h" | 
|  |  | 
|  |  | 
|  | /* | 
|  | *  Encoding of CAPI messages | 
|  | * | 
|  | */ | 
|  |  | 
|  | int capi_conn_req(const char * calledPN, struct sk_buff **skb, int proto) | 
|  | { | 
|  | ushort len; | 
|  |  | 
|  | /* | 
|  | * length | 
|  | *   AppInfoMask - 2 | 
|  | *   BC0         - 3 | 
|  | *   BC1         - 1 | 
|  | *   Chan        - 2 | 
|  | *   Keypad      - 1 | 
|  | *   CPN         - 1 | 
|  | *   CPSA        - 1 | 
|  | *   CalledPN    - 2 + strlen | 
|  | *   CalledPSA   - 1 | 
|  | *   rest...     - 4 | 
|  | *   ---------------- | 
|  | *   Total        18 + strlen | 
|  | */ | 
|  |  | 
|  | len = 18 + strlen(calledPN); | 
|  |  | 
|  | if (proto == ISDN_PROTO_L2_TRANS) | 
|  | len++; | 
|  |  | 
|  | if ((*skb = dev_alloc_skb(len)) == NULL) { | 
|  |  | 
|  | printk(KERN_WARNING "capi_conn_req: alloc_skb failed\n"); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | /* InfoElmMask */ | 
|  | *((ushort*) skb_put(*skb, 2)) = AppInfoMask; | 
|  |  | 
|  | if (proto == ISDN_PROTO_L2_TRANS) | 
|  | { | 
|  | /* Bearer Capability - Mandatory*/ | 
|  | *(skb_put(*skb, 1)) = 3;        /* BC0.Length		*/ | 
|  | *(skb_put(*skb, 1)) = 0x80;     /* Speech		*/ | 
|  | *(skb_put(*skb, 1)) = 0x10;     /* Circuit Mode		*/ | 
|  | *(skb_put(*skb, 1)) = 0x23;     /* A-law		*/ | 
|  | } | 
|  | else | 
|  | { | 
|  | /* Bearer Capability - Mandatory*/ | 
|  | *(skb_put(*skb, 1)) = 2;        /* BC0.Length		*/ | 
|  | *(skb_put(*skb, 1)) = 0x88;     /* Digital Information	*/ | 
|  | *(skb_put(*skb, 1)) = 0x90;     /* BC0.Octect4		*/ | 
|  | } | 
|  |  | 
|  | /* Bearer Capability - Optional*/ | 
|  | *(skb_put(*skb, 1)) = 0;        /* BC1.Length = 0                    */ | 
|  |  | 
|  | *(skb_put(*skb, 1)) = 1;        /* ChannelID.Length = 1              */ | 
|  | *(skb_put(*skb, 1)) = 0x83;     /* Basic Interface - Any Channel     */ | 
|  |  | 
|  | *(skb_put(*skb, 1)) = 0;        /* Keypad.Length = 0                 */ | 
|  |  | 
|  |  | 
|  | *(skb_put(*skb, 1)) = 0;        /* CallingPN.Length = 0              */ | 
|  | *(skb_put(*skb, 1)) = 0;        /* CallingPSA.Length = 0             */ | 
|  |  | 
|  | /* Called Party Number */ | 
|  | *(skb_put(*skb, 1)) = strlen(calledPN) + 1; | 
|  | *(skb_put(*skb, 1)) = 0x81; | 
|  | memcpy(skb_put(*skb, strlen(calledPN)), calledPN, strlen(calledPN)); | 
|  |  | 
|  | /* '#' */ | 
|  |  | 
|  | *(skb_put(*skb, 1)) = 0;       /* CalledPSA.Length = 0     */ | 
|  |  | 
|  | /* LLC.Length  = 0; */ | 
|  | /* HLC0.Length = 0; */ | 
|  | /* HLC1.Length = 0; */ | 
|  | /* UTUS.Length = 0; */ | 
|  | memset(skb_put(*skb, 4), 0, 4); | 
|  |  | 
|  | return len; | 
|  | } | 
|  |  | 
|  | int capi_conn_resp(struct pcbit_chan* chan, struct sk_buff **skb) | 
|  | { | 
|  |  | 
|  | if ((*skb = dev_alloc_skb(5)) == NULL) { | 
|  |  | 
|  | printk(KERN_WARNING "capi_conn_resp: alloc_skb failed\n"); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | *((ushort*) skb_put(*skb, 2) ) = chan->callref; | 
|  | *(skb_put(*skb, 1)) = 0x01;  /* ACCEPT_CALL */ | 
|  | *(skb_put(*skb, 1)) = 0; | 
|  | *(skb_put(*skb, 1)) = 0; | 
|  |  | 
|  | return 5; | 
|  | } | 
|  |  | 
|  | int capi_conn_active_req(struct pcbit_chan* chan, struct sk_buff **skb) | 
|  | { | 
|  | /* | 
|  | * 8 bytes | 
|  | */ | 
|  |  | 
|  | if ((*skb = dev_alloc_skb(8)) == NULL) { | 
|  |  | 
|  | printk(KERN_WARNING "capi_conn_active_req: alloc_skb failed\n"); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | *((ushort*) skb_put(*skb, 2) ) = chan->callref; | 
|  |  | 
|  | #ifdef DEBUG | 
|  | printk(KERN_DEBUG "Call Reference: %04x\n", chan->callref); | 
|  | #endif | 
|  |  | 
|  | *(skb_put(*skb, 1)) = 0;       /*  BC.Length = 0;          */ | 
|  | *(skb_put(*skb, 1)) = 0;       /*  ConnectedPN.Length = 0  */ | 
|  | *(skb_put(*skb, 1)) = 0;       /*  PSA.Length              */ | 
|  | *(skb_put(*skb, 1)) = 0;       /*  LLC.Length = 0;         */ | 
|  | *(skb_put(*skb, 1)) = 0;       /*  HLC.Length = 0;         */ | 
|  | *(skb_put(*skb, 1)) = 0;       /*  UTUS.Length = 0;        */ | 
|  |  | 
|  | return 8; | 
|  | } | 
|  |  | 
|  | int capi_conn_active_resp(struct pcbit_chan* chan, struct sk_buff **skb) | 
|  | { | 
|  | /* | 
|  | * 2 bytes | 
|  | */ | 
|  |  | 
|  | if ((*skb = dev_alloc_skb(2)) == NULL) { | 
|  |  | 
|  | printk(KERN_WARNING "capi_conn_active_resp: alloc_skb failed\n"); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | *((ushort*) skb_put(*skb, 2) ) = chan->callref; | 
|  |  | 
|  | return 2; | 
|  | } | 
|  |  | 
|  |  | 
|  | int capi_select_proto_req(struct pcbit_chan *chan, struct sk_buff **skb, | 
|  | int outgoing) | 
|  | { | 
|  |  | 
|  | /* | 
|  | * 18 bytes | 
|  | */ | 
|  |  | 
|  | if ((*skb = dev_alloc_skb(18)) == NULL) { | 
|  |  | 
|  | printk(KERN_WARNING "capi_select_proto_req: alloc_skb failed\n"); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | *((ushort*) skb_put(*skb, 2) ) = chan->callref; | 
|  |  | 
|  | /* Layer2 protocol */ | 
|  |  | 
|  | switch (chan->proto) { | 
|  | case ISDN_PROTO_L2_X75I: | 
|  | *(skb_put(*skb, 1)) = 0x05;            /* LAPB */ | 
|  | break; | 
|  | case ISDN_PROTO_L2_HDLC: | 
|  | *(skb_put(*skb, 1)) = 0x02; | 
|  | break; | 
|  | case ISDN_PROTO_L2_TRANS: | 
|  | /* | 
|  | *	Voice (a-law) | 
|  | */ | 
|  | *(skb_put(*skb, 1)) = 0x06; | 
|  | break; | 
|  | default: | 
|  | #ifdef DEBUG | 
|  | printk(KERN_DEBUG "Transparent\n"); | 
|  | #endif | 
|  | *(skb_put(*skb, 1)) = 0x03; | 
|  | break; | 
|  | } | 
|  |  | 
|  | *(skb_put(*skb, 1)) = (outgoing ? 0x02 : 0x42);    /* Don't ask */ | 
|  | *(skb_put(*skb, 1)) = 0x00; | 
|  |  | 
|  | *((ushort *) skb_put(*skb, 2)) = MRU; | 
|  |  | 
|  |  | 
|  | *(skb_put(*skb, 1)) = 0x08;           /* Modulo */ | 
|  | *(skb_put(*skb, 1)) = 0x07;           /* Max Window */ | 
|  |  | 
|  | *(skb_put(*skb, 1)) = 0x01;           /* No Layer3 Protocol */ | 
|  |  | 
|  | /* | 
|  | * 2 - layer3 MTU       [10] | 
|  | *   - Modulo           [12] | 
|  | *   - Window | 
|  | *   - layer1 proto     [14] | 
|  | *   - bitrate | 
|  | *   - sub-channel      [16] | 
|  | *   - layer1dataformat [17] | 
|  | */ | 
|  |  | 
|  | memset(skb_put(*skb, 8), 0, 8); | 
|  |  | 
|  | return 18; | 
|  | } | 
|  |  | 
|  |  | 
|  | int capi_activate_transp_req(struct pcbit_chan *chan, struct sk_buff **skb) | 
|  | { | 
|  |  | 
|  | if ((*skb = dev_alloc_skb(7)) == NULL) { | 
|  |  | 
|  | printk(KERN_WARNING "capi_activate_transp_req: alloc_skb failed\n"); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | *((ushort*) skb_put(*skb, 2) ) = chan->callref; | 
|  |  | 
|  |  | 
|  | *(skb_put(*skb, 1)) = chan->layer2link; /* Layer2 id */ | 
|  | *(skb_put(*skb, 1)) = 0x00;             /* Transmit by default */ | 
|  |  | 
|  | *((ushort *) skb_put(*skb, 2)) = MRU; | 
|  |  | 
|  | *(skb_put(*skb, 1)) = 0x01;             /* Enables reception*/ | 
|  |  | 
|  | return 7; | 
|  | } | 
|  |  | 
|  | int capi_tdata_req(struct pcbit_chan* chan, struct sk_buff *skb) | 
|  | { | 
|  | ushort data_len; | 
|  |  | 
|  |  | 
|  | /* | 
|  | * callref      - 2 | 
|  | * layer2link   - 1 | 
|  | * wBlockLength - 2 | 
|  | * data         - 4 | 
|  | * sernum       - 1 | 
|  | */ | 
|  |  | 
|  | data_len = skb->len; | 
|  |  | 
|  | if(skb_headroom(skb) < 10) | 
|  | { | 
|  | printk(KERN_CRIT "No headspace (%u) on headroom %p for capi header\n", skb_headroom(skb), skb); | 
|  | } | 
|  | else | 
|  | { | 
|  | skb_push(skb, 10); | 
|  | } | 
|  |  | 
|  | *((u16 *) (skb->data)) = chan->callref; | 
|  | skb->data[2] = chan->layer2link; | 
|  | *((u16 *) (skb->data + 3)) = data_len; | 
|  |  | 
|  | chan->s_refnum = (chan->s_refnum + 1) % 8; | 
|  | *((u32 *) (skb->data + 5)) = chan->s_refnum; | 
|  |  | 
|  | skb->data[9] = 0;                           /* HDLC frame number */ | 
|  |  | 
|  | return 10; | 
|  | } | 
|  |  | 
|  | int capi_tdata_resp(struct pcbit_chan *chan, struct sk_buff ** skb) | 
|  |  | 
|  | { | 
|  | if ((*skb = dev_alloc_skb(4)) == NULL) { | 
|  |  | 
|  | printk(KERN_WARNING "capi_tdata_resp: alloc_skb failed\n"); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | *((ushort*) skb_put(*skb, 2) ) = chan->callref; | 
|  |  | 
|  | *(skb_put(*skb, 1)) = chan->layer2link; | 
|  | *(skb_put(*skb, 1)) = chan->r_refnum; | 
|  |  | 
|  | return (*skb)->len; | 
|  | } | 
|  |  | 
|  | int capi_disc_req(ushort callref, struct sk_buff **skb, u_char cause) | 
|  | { | 
|  |  | 
|  | if ((*skb = dev_alloc_skb(6)) == NULL) { | 
|  |  | 
|  | printk(KERN_WARNING "capi_disc_req: alloc_skb failed\n"); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | *((ushort*) skb_put(*skb, 2) ) = callref; | 
|  |  | 
|  | *(skb_put(*skb, 1)) = 2;                  /* Cause.Length = 2; */ | 
|  | *(skb_put(*skb, 1)) = 0x80; | 
|  | *(skb_put(*skb, 1)) = 0x80 | cause; | 
|  |  | 
|  | /* | 
|  | * Change it: we should send 'Sic transit gloria Mundi' here ;-) | 
|  | */ | 
|  |  | 
|  | *(skb_put(*skb, 1)) = 0;                   /* UTUS.Length = 0;  */ | 
|  |  | 
|  | return 6; | 
|  | } | 
|  |  | 
|  | int capi_disc_resp(struct pcbit_chan *chan, struct sk_buff **skb) | 
|  | { | 
|  | if ((*skb = dev_alloc_skb(2)) == NULL) { | 
|  |  | 
|  | printk(KERN_WARNING "capi_disc_resp: alloc_skb failed\n"); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | *((ushort*) skb_put(*skb, 2)) = chan->callref; | 
|  |  | 
|  | return 2; | 
|  | } | 
|  |  | 
|  |  | 
|  | /* | 
|  | *  Decoding of CAPI messages | 
|  | * | 
|  | */ | 
|  |  | 
|  | int capi_decode_conn_ind(struct pcbit_chan * chan, | 
|  | struct sk_buff *skb, | 
|  | struct callb_data *info) | 
|  | { | 
|  | int CIlen, len; | 
|  |  | 
|  | /* Call Reference [CAPI] */ | 
|  | chan->callref = *((ushort*) skb->data); | 
|  | skb_pull(skb, 2); | 
|  |  | 
|  | #ifdef DEBUG | 
|  | printk(KERN_DEBUG "Call Reference: %04x\n", chan->callref); | 
|  | #endif | 
|  |  | 
|  | /* Channel Identification */ | 
|  |  | 
|  | /* Expect | 
|  | Len = 1 | 
|  | Octect 3 = 0100 10CC - [ 7 Basic, 4 , 2-1 chan ] | 
|  | */ | 
|  |  | 
|  | CIlen = skb->data[0]; | 
|  | #ifdef DEBUG | 
|  | if (CIlen == 1) { | 
|  |  | 
|  | if ( ((skb->data[1]) & 0xFC) == 0x48 ) | 
|  | printk(KERN_DEBUG "decode_conn_ind: chan ok\n"); | 
|  | printk(KERN_DEBUG "phyChan = %d\n", skb->data[1] & 0x03); | 
|  | } | 
|  | else | 
|  | printk(KERN_DEBUG "conn_ind: CIlen = %d\n", CIlen); | 
|  | #endif | 
|  | skb_pull(skb, CIlen + 1); | 
|  |  | 
|  | /* Calling Party Number */ | 
|  | /* An "additional service" as far as Portugal Telecom is concerned */ | 
|  |  | 
|  | len = skb->data[0]; | 
|  |  | 
|  | if (len > 0) { | 
|  | int count = 1; | 
|  |  | 
|  | #ifdef DEBUG | 
|  | printk(KERN_DEBUG "CPN: Octect 3 %02x\n", skb->data[1]); | 
|  | #endif | 
|  | if ((skb->data[1] & 0x80) == 0) | 
|  | count = 2; | 
|  |  | 
|  | if (!(info->data.setup.CallingPN = kmalloc(len - count + 1, GFP_ATOMIC))) | 
|  | return -1; | 
|  |  | 
|  | skb_copy_from_linear_data_offset(skb, count + 1, | 
|  | info->data.setup.CallingPN, | 
|  | len - count); | 
|  | info->data.setup.CallingPN[len - count] = 0; | 
|  |  | 
|  | } | 
|  | else { | 
|  | info->data.setup.CallingPN = NULL; | 
|  | printk(KERN_DEBUG "NULL CallingPN\n"); | 
|  | } | 
|  |  | 
|  | skb_pull(skb, len + 1); | 
|  |  | 
|  | /* Calling Party Subaddress */ | 
|  | skb_pull(skb, skb->data[0] + 1); | 
|  |  | 
|  | /* Called Party Number */ | 
|  |  | 
|  | len = skb->data[0]; | 
|  |  | 
|  | if (len > 0) { | 
|  | int count = 1; | 
|  |  | 
|  | if ((skb->data[1] & 0x80) == 0) | 
|  | count = 2; | 
|  |  | 
|  | if (!(info->data.setup.CalledPN = kmalloc(len - count + 1, GFP_ATOMIC))) | 
|  | return -1; | 
|  |  | 
|  | skb_copy_from_linear_data_offset(skb, count + 1, | 
|  | info->data.setup.CalledPN, | 
|  | len - count); | 
|  | info->data.setup.CalledPN[len - count] = 0; | 
|  |  | 
|  | } | 
|  | else { | 
|  | info->data.setup.CalledPN = NULL; | 
|  | printk(KERN_DEBUG "NULL CalledPN\n"); | 
|  | } | 
|  |  | 
|  | skb_pull(skb, len + 1); | 
|  |  | 
|  | /* Called Party Subaddress */ | 
|  | skb_pull(skb, skb->data[0] + 1); | 
|  |  | 
|  | /* LLC */ | 
|  | skb_pull(skb, skb->data[0] + 1); | 
|  |  | 
|  | /* HLC */ | 
|  | skb_pull(skb, skb->data[0] + 1); | 
|  |  | 
|  | /* U2U */ | 
|  | skb_pull(skb, skb->data[0] + 1); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* | 
|  | *  returns errcode | 
|  | */ | 
|  |  | 
|  | int capi_decode_conn_conf(struct pcbit_chan * chan, struct sk_buff *skb, | 
|  | int *complete) | 
|  | { | 
|  | int errcode; | 
|  |  | 
|  | chan->callref = *((ushort *) skb->data);     /* Update CallReference */ | 
|  | skb_pull(skb, 2); | 
|  |  | 
|  | errcode = *((ushort *) skb->data);   /* read errcode */ | 
|  | skb_pull(skb, 2); | 
|  |  | 
|  | *complete = *(skb->data); | 
|  | skb_pull(skb, 1); | 
|  |  | 
|  | /* FIX ME */ | 
|  | /* This is actually a firmware bug */ | 
|  | if (!*complete) | 
|  | { | 
|  | printk(KERN_DEBUG "complete=%02x\n", *complete); | 
|  | *complete = 1; | 
|  | } | 
|  |  | 
|  |  | 
|  | /* Optional Bearer Capability */ | 
|  | skb_pull(skb, *(skb->data) + 1); | 
|  |  | 
|  | /* Channel Identification */ | 
|  | skb_pull(skb, *(skb->data) + 1); | 
|  |  | 
|  | /* High Layer Compatibility follows */ | 
|  | skb_pull(skb, *(skb->data) + 1); | 
|  |  | 
|  | return errcode; | 
|  | } | 
|  |  | 
|  | int capi_decode_conn_actv_ind(struct pcbit_chan * chan, struct sk_buff *skb) | 
|  | { | 
|  | ushort len; | 
|  | #ifdef DEBUG | 
|  | char str[32]; | 
|  | #endif | 
|  |  | 
|  | /* Yet Another Bearer Capability */ | 
|  | skb_pull(skb, *(skb->data) + 1); | 
|  |  | 
|  |  | 
|  | /* Connected Party Number */ | 
|  | len=*(skb->data); | 
|  |  | 
|  | #ifdef DEBUG | 
|  | if (len > 1 && len < 31) { | 
|  | skb_copy_from_linear_data_offset(skb, 2, str, len - 1); | 
|  | str[len] = 0; | 
|  | printk(KERN_DEBUG "Connected Party Number: %s\n", str); | 
|  | } | 
|  | else | 
|  | printk(KERN_DEBUG "actv_ind CPN len = %d\n", len); | 
|  | #endif | 
|  |  | 
|  | skb_pull(skb, len + 1); | 
|  |  | 
|  | /* Connected Subaddress */ | 
|  | skb_pull(skb, *(skb->data) + 1); | 
|  |  | 
|  | /* Low Layer Capability */ | 
|  | skb_pull(skb, *(skb->data) + 1); | 
|  |  | 
|  | /* High Layer Capability */ | 
|  | skb_pull(skb, *(skb->data) + 1); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | int capi_decode_conn_actv_conf(struct pcbit_chan * chan, struct sk_buff *skb) | 
|  | { | 
|  | ushort errcode; | 
|  |  | 
|  | errcode = *((ushort*) skb->data); | 
|  | skb_pull(skb, 2); | 
|  |  | 
|  | /* Channel Identification | 
|  | skb_pull(skb, skb->data[0] + 1); | 
|  | */ | 
|  | return errcode; | 
|  | } | 
|  |  | 
|  |  | 
|  | int capi_decode_sel_proto_conf(struct pcbit_chan *chan, struct sk_buff *skb) | 
|  | { | 
|  | ushort errcode; | 
|  |  | 
|  | chan->layer2link = *(skb->data); | 
|  | skb_pull(skb, 1); | 
|  |  | 
|  | errcode = *((ushort*) skb->data); | 
|  | skb_pull(skb, 2); | 
|  |  | 
|  | return errcode; | 
|  | } | 
|  |  | 
|  | int capi_decode_actv_trans_conf(struct pcbit_chan *chan, struct sk_buff *skb) | 
|  | { | 
|  | ushort errcode; | 
|  |  | 
|  | if (chan->layer2link != *(skb->data) ) | 
|  | printk("capi_decode_actv_trans_conf: layer2link doesn't match\n"); | 
|  |  | 
|  | skb_pull(skb, 1); | 
|  |  | 
|  | errcode = *((ushort*) skb->data); | 
|  | skb_pull(skb, 2); | 
|  |  | 
|  | return errcode; | 
|  | } | 
|  |  | 
|  | int capi_decode_disc_ind(struct pcbit_chan *chan, struct sk_buff *skb) | 
|  | { | 
|  | ushort len; | 
|  | #ifdef DEBUG | 
|  | int i; | 
|  | #endif | 
|  | /* Cause */ | 
|  |  | 
|  | len = *(skb->data); | 
|  | skb_pull(skb, 1); | 
|  |  | 
|  | #ifdef DEBUG | 
|  |  | 
|  | for (i=0; i<len; i++) | 
|  | printk(KERN_DEBUG "Cause Octect %d: %02x\n", i+3, | 
|  | *(skb->data + i)); | 
|  | #endif | 
|  |  | 
|  | skb_pull(skb, len); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | #ifdef DEBUG | 
|  | int capi_decode_debug_188(u_char *hdr, ushort hdrlen) | 
|  | { | 
|  | char str[64]; | 
|  | int len; | 
|  |  | 
|  | len = hdr[0]; | 
|  |  | 
|  | if (len < 64 && len == hdrlen - 1) { | 
|  | memcpy(str, hdr + 1, hdrlen - 1); | 
|  | str[hdrlen - 1] = 0; | 
|  | printk("%s\n", str); | 
|  | } | 
|  | else | 
|  | printk("debug message incorrect\n"); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  | #endif | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  |  |