Annotation of XNU/bsd/netccitt/pk_input.c, revision 1.1.1.1

1.1       root        1: /*
                      2:  * Copyright (c) 2000 Apple Computer, Inc. All rights reserved.
                      3:  *
                      4:  * @APPLE_LICENSE_HEADER_START@
                      5:  * 
                      6:  * The contents of this file constitute Original Code as defined in and
                      7:  * are subject to the Apple Public Source License Version 1.1 (the
                      8:  * "License").  You may not use this file except in compliance with the
                      9:  * License.  Please obtain a copy of the License at
                     10:  * http://www.apple.com/publicsource and read it before using this file.
                     11:  * 
                     12:  * This Original Code and all software distributed under the License are
                     13:  * distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER
                     14:  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
                     15:  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
                     16:  * FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT.  Please see the
                     17:  * License for the specific language governing rights and limitations
                     18:  * under the License.
                     19:  * 
                     20:  * @APPLE_LICENSE_HEADER_END@
                     21:  */
                     22: /*
                     23:  * Copyright (c) University of British Columbia, 1984
                     24:  * Copyright (C) Computer Science Department IV, 
                     25:  *              University of Erlangen-Nuremberg, Germany, 1992
                     26:  * Copyright (c) 1991, 1992, 1993
                     27:  *     The Regents of the University of California.  All rights reserved.
                     28:  *
                     29:  * This code is derived from software contributed to Berkeley by the
                     30:  * Laboratory for Computation Vision and the Computer Science Department
                     31:  * of the the University of British Columbia and the Computer Science
                     32:  * Department (IV) of the University of Erlangen-Nuremberg, Germany.
                     33:  *
                     34:  * Redistribution and use in source and binary forms, with or without
                     35:  * modification, are permitted provided that the following conditions
                     36:  * are met:
                     37:  * 1. Redistributions of source code must retain the above copyright
                     38:  *    notice, this list of conditions and the following disclaimer.
                     39:  * 2. Redistributions in binary form must reproduce the above copyright
                     40:  *    notice, this list of conditions and the following disclaimer in the
                     41:  *    documentation and/or other materials provided with the distribution.
                     42:  * 3. All advertising materials mentioning features or use of this software
                     43:  *    must display the following acknowledgement:
                     44:  *     This product includes software developed by the University of
                     45:  *     California, Berkeley and its contributors.
                     46:  * 4. Neither the name of the University nor the names of its contributors
                     47:  *    may be used to endorse or promote products derived from this software
                     48:  *    without specific prior written permission.
                     49:  *
                     50:  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
                     51:  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
                     52:  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
                     53:  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
                     54:  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
                     55:  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
                     56:  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
                     57:  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
                     58:  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
                     59:  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
                     60:  * SUCH DAMAGE.
                     61:  *
                     62:  *     @(#)pk_input.c  8.1 (Berkeley) 6/10/93
                     63:  */
                     64: 
                     65: #include <sys/param.h>
                     66: #include <sys/systm.h>
                     67: #include <sys/mbuf.h>
                     68: #include <sys/socket.h>
                     69: #include <sys/protosw.h>
                     70: #include <sys/socketvar.h>
                     71: #include <sys/errno.h>
                     72: #include <sys/malloc.h>
                     73: 
                     74: #include <net/if.h>
                     75: #include <net/if_dl.h>
                     76: #include <net/if_llc.h>
                     77: #include <net/route.h>
                     78: 
                     79: #include <netccitt/dll.h>
                     80: #include <netccitt/x25.h>
                     81: #include <netccitt/pk.h>
                     82: #include <netccitt/pk_var.h>
                     83: #include <netccitt/llc_var.h>
                     84: 
                     85: struct pkcb_q pkcb_q = {&pkcb_q, &pkcb_q};
                     86: 
                     87: /*
                     88:  * ccittintr() is the generic interrupt handler for HDLC, LLC2, and X.25. This
                     89:  * allows to have kernel running X.25 but no HDLC or LLC2 or both (in case we
                     90:  * employ boards that do all the stuff themselves, e.g. ADAX X.25 or TPS ISDN.)
                     91:  */
                     92: void
                     93: ccittintr ()
                     94: {
                     95:        extern struct ifqueue pkintrq;
                     96:        extern struct ifqueue hdintrq;
                     97:        extern struct ifqueue llcintrq;
                     98: 
                     99: #if HDLC
                    100:        if (hdintrq.ifq_len)
                    101:                hdintr ();
                    102: #endif
                    103: #if LLC
                    104:        if (llcintrq.ifq_len)
                    105:                llcintr ();
                    106: #endif
                    107:        if (pkintrq.ifq_len)
                    108:                pkintr ();
                    109: }
                    110: 
                    111: struct pkcb *
                    112: pk_newlink (ia, llnext)
                    113: struct x25_ifaddr *ia;
                    114: caddr_t llnext;
                    115: {
                    116:        register struct x25config *xcp = &ia -> ia_xc;
                    117:        register struct pkcb *pkp;
                    118:        register struct pklcd *lcp;
                    119:        register struct protosw *pp;
                    120:        unsigned size;
                    121: 
                    122:        pp = pffindproto (AF_CCITT, (int) xcp -> xc_lproto, 0);
                    123:        if (pp == 0 || pp -> pr_output == 0) {
                    124:                pk_message (0, xcp, "link level protosw error");
                    125:                return ((struct pkcb *)0);
                    126:        }
                    127:        /*
                    128:         * Allocate a network control block structure
                    129:         */
                    130:        size = sizeof (struct pkcb);
                    131: //     pkp = (struct pkcb *) malloc (size, M_PCB, M_WAITOK);
                    132:        MALLOC(pkp, struct pkcb *, size, M_PCB, M_WAITOK);
                    133:        if (pkp == 0)
                    134:                return ((struct pkcb *)0);
                    135:        bzero ((caddr_t) pkp, size);
                    136:        pkp -> pk_lloutput = pp -> pr_output;
                    137:        pkp -> pk_llctlinput = (caddr_t (*)()) pp -> pr_ctlinput;
                    138:        pkp -> pk_xcp = xcp;
                    139:        pkp -> pk_ia = ia;
                    140:        pkp -> pk_state = DTE_WAITING;
                    141:        pkp -> pk_llnext = llnext;
                    142:        insque (pkp, &pkcb_q);
                    143: 
                    144:        /*
                    145:         * set defaults
                    146:         */
                    147: 
                    148:        if (xcp -> xc_pwsize == 0)
                    149:                xcp -> xc_pwsize = DEFAULT_WINDOW_SIZE;
                    150:        if (xcp -> xc_psize == 0)
                    151:                xcp -> xc_psize = X25_PS128;
                    152:        /*
                    153:         * Allocate logical channel descriptor vector
                    154:         */
                    155: 
                    156:        (void) pk_resize (pkp);
                    157:        return (pkp);
                    158: }
                    159: 
                    160: 
                    161: pk_dellink (pkp)
                    162: register struct pkcb *pkp;
                    163: {
                    164:        register int i;
                    165:        register struct protosw *pp;
                    166:        
                    167:        /*
                    168:         * Essentially we have the choice to
                    169:         * (a) go ahead and let the route be deleted and
                    170:         *     leave the pkcb associated with that route
                    171:         *     as it is, i.e. the connections stay open
                    172:         * (b) do a pk_disconnect() on all channels associated
                    173:         *     with the route via the pkcb and then proceed.
                    174:         *
                    175:         * For the time being we stick with (b)
                    176:         */
                    177:        
                    178:        for (i = 1; i < pkp -> pk_maxlcn; ++i)
                    179:                if (pkp -> pk_chan[i])
                    180:                        pk_disconnect (pkp -> pk_chan[i]);
                    181: 
                    182:        /*
                    183:         * Free the pkcb
                    184:         */
                    185: 
                    186:        /*
                    187:         * First find the protoswitch to get hold of the link level
                    188:         * protocol to be notified that the packet level entity is
                    189:         * dissolving ...
                    190:         */
                    191:        pp = pffindproto (AF_CCITT, (int) pkp -> pk_xcp -> xc_lproto, 0);
                    192:        if (pp == 0 || pp -> pr_output == 0) {
                    193:                pk_message (0, pkp -> pk_xcp, "link level protosw error");
                    194:                return (EPROTONOSUPPORT);
                    195:        }
                    196: 
                    197:        pkp -> pk_refcount--;
                    198:        if (!pkp -> pk_refcount) {
                    199:                struct dll_ctlinfo ctlinfo;
                    200: 
                    201:                remque (pkp);
                    202:                if (pkp -> pk_rt -> rt_llinfo == (caddr_t) pkp)
                    203:                        pkp -> pk_rt -> rt_llinfo = (caddr_t) NULL;
                    204:                
                    205:                /*
                    206:                 * Tell the link level that the pkcb is dissolving
                    207:                 */
                    208:                if (pp -> pr_ctlinput && pkp -> pk_llnext) {
                    209:                        ctlinfo.dlcti_pcb = pkp -> pk_llnext;
                    210:                        ctlinfo.dlcti_rt = pkp -> pk_rt;
                    211:                        (pp -> pr_ctlinput)(PRC_DISCONNECT_REQUEST, 
                    212:                                            pkp -> pk_xcp, &ctlinfo);
                    213:                }
                    214:                FREE((caddr_t) pkp -> pk_chan, M_IFADDR);
                    215:                FREE((caddr_t) pkp, M_PCB);
                    216:        }
                    217: 
                    218:        return (0);
                    219: }
                    220: 
                    221: 
                    222: pk_resize (pkp)
                    223: register struct pkcb *pkp;
                    224: {
                    225:        struct pklcd *dev_lcp = 0;
                    226:        struct x25config *xcp = pkp -> pk_xcp;
                    227:        if (pkp -> pk_chan &&
                    228:            (pkp -> pk_maxlcn != xcp -> xc_maxlcn)) {
                    229:                pk_restart (pkp, X25_RESTART_NETWORK_CONGESTION);
                    230:                dev_lcp = pkp -> pk_chan[0];
                    231:                FREE((caddr_t) pkp -> pk_chan, M_IFADDR);
                    232:                pkp -> pk_chan = 0;
                    233:        }
                    234:        if (pkp -> pk_chan == 0) {
                    235:                unsigned size;
                    236:                pkp -> pk_maxlcn = xcp -> xc_maxlcn;
                    237:                size = (pkp -> pk_maxlcn + 1) * sizeof (struct pklcd *);
                    238: //             pkp -> pk_chan =
                    239: //                     (struct pklcd **) malloc (size, M_IFADDR, M_WAITOK);
                    240:                MALLOC(pkp->pk_chan, struct pklcd **, size, M_IFADDR, M_WAITOK);
                    241:                if (pkp -> pk_chan) {
                    242:                        bzero ((caddr_t) pkp -> pk_chan, size);
                    243:                        /*
                    244:                         * Allocate a logical channel descriptor for lcn 0
                    245:                         */
                    246:                        if (dev_lcp == 0 &&
                    247:                            (dev_lcp = pk_attach ((struct socket *)0)) == 0)
                    248:                                return (ENOBUFS);
                    249:                        dev_lcp -> lcd_state = READY;
                    250:                        dev_lcp -> lcd_pkp = pkp;
                    251:                        pkp -> pk_chan[0] = dev_lcp;
                    252:                } else {
                    253:                        if (dev_lcp)
                    254:                                pk_close (dev_lcp);
                    255:                        return (ENOBUFS);
                    256:                }
                    257:        }
                    258:        return 0;
                    259: }
                    260: 
                    261: /* 
                    262:  *  This procedure is called by the link level whenever the link
                    263:  *  becomes operational, is reset, or when the link goes down. 
                    264:  */
                    265: /*VARARGS*/
                    266: caddr_t
                    267: pk_ctlinput (code, src, addr)
                    268:        int code;
                    269:        struct sockaddr *src;
                    270:        caddr_t addr;
                    271: {
                    272:        register struct pkcb *pkp = (struct pkcb *) addr;
                    273: 
                    274:        switch (code) {
                    275:        case PRC_LINKUP: 
                    276:                if (pkp -> pk_state == DTE_WAITING)
                    277:                        pk_restart (pkp, X25_RESTART_NETWORK_CONGESTION);
                    278:                break;
                    279: 
                    280:        case PRC_LINKDOWN: 
                    281:                pk_restart (pkp, -1);   /* Clear all active circuits */
                    282:                pkp -> pk_state = DTE_WAITING;
                    283:                break;
                    284: 
                    285:        case PRC_LINKRESET: 
                    286:                pk_restart (pkp, X25_RESTART_NETWORK_CONGESTION);
                    287:                break;
                    288:                
                    289:        case PRC_CONNECT_INDICATION: {
                    290:                struct rtentry *llrt;
                    291: 
                    292:                if ((llrt = rtalloc1(src, 0)) == 0)
                    293:                        return 0;
                    294:                else llrt -> rt_refcnt--;
                    295:                
                    296:                pkp = (((struct npaidbentry *) llrt -> rt_llinfo) -> np_rt) ?
                    297:                        (struct pkcb *)(((struct npaidbentry *) llrt -> rt_llinfo) -> np_rt -> rt_llinfo) : (struct pkcb *) 0;
                    298:                if (pkp == (struct pkcb *) 0)
                    299:                        return 0;
                    300:                pkp -> pk_llnext = addr;
                    301: 
                    302:                return ((caddr_t) pkp);
                    303:        }
                    304:        case PRC_DISCONNECT_INDICATION:
                    305:                pk_restart (pkp, -1) ;  /* Clear all active circuits */
                    306:                pkp -> pk_state = DTE_WAITING;
                    307:                pkp -> pk_llnext = (caddr_t) 0;
                    308:        }
                    309:        return (0);
                    310: }
                    311: struct ifqueue pkintrq;
                    312: /*
                    313:  * This routine is called if there are semi-smart devices that do HDLC
                    314:  * in hardware and want to queue the packet and call level 3 directly
                    315:  */
                    316: pkintr ()
                    317: {
                    318:        register struct mbuf *m;
                    319:        register struct ifaddr *ifa;
                    320:        register struct ifnet *ifp;
                    321:        register int s;
                    322: 
                    323:        for (;;) {
                    324:                s = splimp ();
                    325:                IF_DEQUEUE (&pkintrq, m);
                    326:                splx (s);
                    327:                if (m == 0)
                    328:                        break;
                    329:                if (m -> m_len < PKHEADERLN) {
                    330:                        printf ("pkintr: packet too short (len=%d)\n",
                    331:                                m -> m_len);
                    332:                        m_freem (m);
                    333:                        continue;
                    334:                }
                    335:                pk_input (m);
                    336:        }
                    337: }
                    338: struct mbuf *pk_bad_packet;
                    339: struct mbuf_cache pk_input_cache = {0 };
                    340: /* 
                    341:  *  X.25 PACKET INPUT
                    342:  *
                    343:  *  This procedure is called by a link level procedure whenever
                    344:  *  an information frame is received. It decodes the packet and
                    345:  *  demultiplexes based on the logical channel number.
                    346:  *
                    347:  *  We change the original conventions of the UBC code here --
                    348:  *  since there may be multiple pkcb's for a given interface
                    349:  *  of type 802.2 class 2, we retrieve which one it is from
                    350:  *  m_pkthdr.rcvif (which has been overwritten by lower layers);
                    351:  *  That field is then restored for the benefit of upper layers which
                    352:  *  may make use of it, such as CLNP.
                    353:  *
                    354:  */
                    355: 
                    356: #define RESTART_DTE_ORIGINATED(xp) (((xp) -> packet_cause == X25_RESTART_DTE_ORIGINATED) || \
                    357:                            ((xp) -> packet_cause >= X25_RESTART_DTE_ORIGINATED2))
                    358: 
                    359: pk_input (m)
                    360: register struct mbuf *m;
                    361: {
                    362:        register struct x25_packet *xp;
                    363:        register struct pklcd *lcp;
                    364:        register struct socket *so = 0;
                    365:        register struct pkcb *pkp;
                    366:        int  ptype, lcn, lcdstate = LISTEN;
                    367: 
                    368:        if (pk_input_cache.mbc_size || pk_input_cache.mbc_oldsize)
                    369:                mbuf_cache (&pk_input_cache, m);
                    370:        if ((m -> m_flags & M_PKTHDR) == 0)
                    371:                panic ("pkintr");
                    372: 
                    373:        if ((pkp = (struct pkcb *) m -> m_pkthdr.rcvif) == 0)
                    374:                return;
                    375:        xp = mtod (m, struct x25_packet *);
                    376:        ptype = pk_decode (xp);
                    377:        lcn = LCN(xp);
                    378:        lcp = pkp -> pk_chan[lcn];
                    379: 
                    380:        /* 
                    381:         *  If the DTE is in Restart  state, then it will ignore data, 
                    382:         *  interrupt, call setup and clearing, flow control and reset 
                    383:         *  packets.
                    384:         */
                    385:        if (lcn < 0 || lcn > pkp -> pk_maxlcn) {
                    386:                pk_message (lcn, pkp -> pk_xcp, "illegal lcn");
                    387:                m_freem (m);
                    388:                return;
                    389:        }
                    390: 
                    391:        pk_trace (pkp -> pk_xcp, m, "P-In");
                    392: 
                    393:        if (pkp -> pk_state != DTE_READY && ptype != RESTART && ptype != RESTART_CONF) {
                    394:                m_freem (m);
                    395:                return;
                    396:        }
                    397:        if (lcp) {
                    398:                so = lcp -> lcd_so;
                    399:                lcdstate = lcp -> lcd_state;
                    400:        } else {
                    401:                if (ptype == CLEAR) {   /* idle line probe (Datapac specific) */
                    402:                        /* send response on lcd 0's output queue */
                    403:                        lcp = pkp -> pk_chan[0];
                    404:                        lcp -> lcd_template = pk_template (lcn, X25_CLEAR_CONFIRM);
                    405:                        pk_output (lcp);
                    406:                        m_freem (m);
                    407:                        return;
                    408:                }
                    409:                if (ptype != CALL)
                    410:                        ptype = INVALID_PACKET;
                    411:        }
                    412: 
                    413:        if (lcn == 0 && ptype != RESTART && ptype != RESTART_CONF) {
                    414:                pk_message (0, pkp -> pk_xcp, "illegal ptype (%d, %s) on lcn 0",
                    415:                        ptype, pk_name[ptype / MAXSTATES]);
                    416:                if (pk_bad_packet)
                    417:                        m_freem (pk_bad_packet);
                    418:                pk_bad_packet = m;
                    419:                return;
                    420:        }
                    421: 
                    422:        m -> m_pkthdr.rcvif = pkp -> pk_ia -> ia_ifp;
                    423: 
                    424:        switch (ptype + lcdstate) {
                    425:        /* 
                    426:         *  Incoming Call packet received. 
                    427:         */
                    428:        case CALL + LISTEN: 
                    429:                pk_incoming_call (pkp, m);
                    430:                break;
                    431: 
                    432:        /*      
                    433:         *  Call collision: Just throw this "incoming call" away since 
                    434:         *  the DCE will ignore it anyway. 
                    435:         */
                    436:        case CALL + SENT_CALL: 
                    437:                pk_message ((int) lcn, pkp -> pk_xcp, 
                    438:                        "incoming call collision");
                    439:                break;
                    440: 
                    441:        /* 
                    442:         *  Call confirmation packet received. This usually means our
                    443:         *  previous connect request is now complete.
                    444:         */
                    445:        case CALL_ACCEPTED + SENT_CALL: 
                    446:                MCHTYPE(m, MT_CONTROL);
                    447:                pk_call_accepted (lcp, m);
                    448:                break;
                    449: 
                    450:        /* 
                    451:         *  This condition can only happen if the previous state was
                    452:         *  SENT_CALL. Just ignore the packet, eventually a clear 
                    453:         *  confirmation should arrive.
                    454:         */
                    455:        case CALL_ACCEPTED + SENT_CLEAR: 
                    456:                break;
                    457: 
                    458:        /* 
                    459:         *  Clear packet received. This requires a complete tear down
                    460:         *  of the virtual circuit.  Free buffers and control blocks.
                    461:         *  and send a clear confirmation.
                    462:         */
                    463:        case CLEAR + READY:
                    464:        case CLEAR + RECEIVED_CALL: 
                    465:        case CLEAR + SENT_CALL: 
                    466:        case CLEAR + DATA_TRANSFER: 
                    467:                lcp -> lcd_state = RECEIVED_CLEAR;
                    468:                lcp -> lcd_template = pk_template (lcp -> lcd_lcn, X25_CLEAR_CONFIRM);
                    469:                pk_output (lcp);
                    470:                pk_clearcause (pkp, xp);
                    471:                if (lcp -> lcd_upper) {
                    472:                        MCHTYPE(m, MT_CONTROL);
                    473:                        lcp -> lcd_upper (lcp, m);
                    474:                }
                    475:                pk_close (lcp);
                    476:                lcp = 0;
                    477:                break;
                    478: 
                    479:        /* 
                    480:         *  Clear collision: Treat this clear packet as a confirmation.
                    481:         */
                    482:        case CLEAR + SENT_CLEAR: 
                    483:                pk_close (lcp);
                    484:                break;
                    485: 
                    486:        /* 
                    487:         *  Clear confirmation received. This usually means the virtual
                    488:         *  circuit is now completely removed.
                    489:         */
                    490:        case CLEAR_CONF + SENT_CLEAR: 
                    491:                pk_close (lcp);
                    492:                break;
                    493: 
                    494:        /* 
                    495:         *  A clear confirmation on an unassigned logical channel - just
                    496:         *  ignore it. Note: All other packets on an unassigned channel
                    497:         *  results in a clear.
                    498:         */
                    499:        case CLEAR_CONF + READY:
                    500:        case CLEAR_CONF + LISTEN:
                    501:                break;
                    502: 
                    503:        /* 
                    504:         *  Data packet received. Pass on to next level. Move the Q and M
                    505:         *  bits into the data portion for the next level.
                    506:         */
                    507:        case DATA + DATA_TRANSFER: 
                    508:                if (lcp -> lcd_reset_condition) {
                    509:                        ptype = DELETE_PACKET;
                    510:                        break;
                    511:                }
                    512: 
                    513:                /* 
                    514:                 *  Process the P(S) flow control information in this Data packet. 
                    515:                 *  Check that the packets arrive in the correct sequence and that 
                    516:                 *  they are within the "lcd_input_window". Input window rotation is 
                    517:                 *  initiated by the receive interface.
                    518:                 */
                    519: 
                    520:                if (PS(xp) != ((lcp -> lcd_rsn + 1) % MODULUS) ||
                    521:                        PS(xp) == ((lcp -> lcd_input_window + lcp -> lcd_windowsize) % MODULUS)) {
                    522:                        m_freem (m);
                    523:                        pk_procerror (RESET, lcp, "p(s) flow control error", 1);
                    524:                        break;
                    525:                }
                    526:                lcp -> lcd_rsn = PS(xp);
                    527: 
                    528:                if (pk_ack (lcp, PR(xp)) != PACKET_OK) {
                    529:                        m_freem (m);
                    530:                        break;
                    531:                }
                    532:                m -> m_data += PKHEADERLN;
                    533:                m -> m_len -= PKHEADERLN;
                    534:                m -> m_pkthdr.len -= PKHEADERLN;
                    535: 
                    536:                lcp -> lcd_rxcnt++;
                    537:                if (lcp -> lcd_flags & X25_MBS_HOLD) {
                    538:                        register struct mbuf *n = lcp -> lcd_cps;
                    539:                        int mbit = MBIT(xp);
                    540:                        octet q_and_d_bits;
                    541: 
                    542:                        if (n) {
                    543:                                n -> m_pkthdr.len += m -> m_pkthdr.len;
                    544:                                while (n -> m_next)
                    545:                                        n = n -> m_next;
                    546:                                n -> m_next = m;
                    547:                                m = lcp -> lcd_cps;
                    548: 
                    549:                                if (lcp -> lcd_cpsmax &&
                    550:                                    n -> m_pkthdr.len > lcp -> lcd_cpsmax) {
                    551:                                        pk_procerror (RESET, lcp,
                    552:                                                "C.P.S. overflow", 128);
                    553:                                        return;
                    554:                                }
                    555:                                q_and_d_bits = 0xc0 & *(octet *) xp;
                    556:                                xp = (struct x25_packet *)
                    557:                                        (mtod (m, octet *) - PKHEADERLN);
                    558:                                *(octet *) xp |= q_and_d_bits;
                    559:                        }
                    560:                        if (mbit) {
                    561:                                lcp -> lcd_cps = m;
                    562:                                pk_flowcontrol (lcp, 0, 1);
                    563:                                return;
                    564:                        }
                    565:                        lcp -> lcd_cps = 0;
                    566:                }
                    567:                if (so == 0)
                    568:                        break;
                    569:                if (lcp -> lcd_flags & X25_MQBIT) {
                    570:                        octet t = (X25GBITS(xp -> bits, q_bit)) ? t = 0x80 : 0;
                    571: 
                    572:                        if (MBIT(xp))
                    573:                                t |= 0x40;
                    574:                        m -> m_data -= 1;
                    575:                        m -> m_len += 1;
                    576:                        m -> m_pkthdr.len += 1;
                    577:                        *mtod (m, octet *) = t;
                    578:                }
                    579: 
                    580:                /*
                    581:                 * Discard Q-BIT packets if the application
                    582:                 * doesn't want to be informed of M and Q bit status
                    583:                 */
                    584:                if (X25GBITS(xp -> bits, q_bit) 
                    585:                    && (lcp -> lcd_flags & X25_MQBIT) == 0) {
                    586:                        m_freem (m);
                    587:                        /*
                    588:                         * NB.  This is dangerous: sending a RR here can
                    589:                         * cause sequence number errors if a previous data
                    590:                         * packet has not yet been passed up to the application
                    591:                         * (RR's are normally generated via PRU_RCVD).
                    592:                         */
                    593:                        pk_flowcontrol (lcp, 0, 1);
                    594:                } else {
                    595:                        sbappendrecord (&so -> so_rcv, m);
                    596:                        sorwakeup (so);
                    597:                }
                    598:                break;
                    599: 
                    600:        /* 
                    601:         *  Interrupt packet received.
                    602:         */
                    603:        case INTERRUPT + DATA_TRANSFER: 
                    604:                if (lcp -> lcd_reset_condition)
                    605:                        break;
                    606:                lcp -> lcd_intrdata = xp -> packet_data;
                    607:                lcp -> lcd_template = pk_template (lcp -> lcd_lcn, X25_INTERRUPT_CONFIRM);
                    608:                pk_output (lcp);
                    609:                m -> m_data += PKHEADERLN;
                    610:                m -> m_len -= PKHEADERLN;
                    611:                m -> m_pkthdr.len -= PKHEADERLN;
                    612:                MCHTYPE(m, MT_OOBDATA);
                    613:                if (so) {
                    614:                        if (so -> so_options & SO_OOBINLINE)
                    615:                                sbinsertoob (&so -> so_rcv, m);
                    616:                        else
                    617:                                m_freem (m);
                    618:                        sohasoutofband (so);
                    619:                }
                    620:                break;
                    621: 
                    622:        /* 
                    623:         *  Interrupt confirmation packet received.
                    624:         */
                    625:        case INTERRUPT_CONF + DATA_TRANSFER: 
                    626:                if (lcp -> lcd_reset_condition)
                    627:                        break;
                    628:                if (lcp -> lcd_intrconf_pending == TRUE)
                    629:                        lcp -> lcd_intrconf_pending = FALSE;
                    630:                else
                    631:                        pk_procerror (RESET, lcp, "unexpected packet", 43);
                    632:                break;
                    633: 
                    634:        /* 
                    635:         *  Receiver ready received. Rotate the output window and output
                    636:         *  any data packets waiting transmission.
                    637:         */
                    638:        case RR + DATA_TRANSFER: 
                    639:                if (lcp -> lcd_reset_condition ||
                    640:                    pk_ack (lcp, PR(xp)) != PACKET_OK) {
                    641:                        ptype = DELETE_PACKET;
                    642:                        break;
                    643:                }
                    644:                if (lcp -> lcd_rnr_condition == TRUE)
                    645:                        lcp -> lcd_rnr_condition = FALSE;
                    646:                pk_output (lcp);
                    647:                break;
                    648: 
                    649:        /* 
                    650:         *  Receiver Not Ready received. Packets up to the P(R) can be
                    651:         *  be sent. Condition is cleared with a RR.
                    652:         */
                    653:        case RNR + DATA_TRANSFER: 
                    654:                if (lcp -> lcd_reset_condition ||
                    655:                    pk_ack (lcp, PR(xp)) != PACKET_OK) {
                    656:                        ptype = DELETE_PACKET;
                    657:                        break;
                    658:                }
                    659:                lcp -> lcd_rnr_condition = TRUE;
                    660:                break;
                    661: 
                    662:        /* 
                    663:         *  Reset packet received. Set state to FLOW_OPEN.  The Input and
                    664:         *  Output window edges ar set to zero. Both the send and receive
                    665:         *  numbers are reset. A confirmation is returned.
                    666:         */
                    667:        case RESET + DATA_TRANSFER: 
                    668:                if (lcp -> lcd_reset_condition)
                    669:                        /* Reset collision. Just ignore packet. */
                    670:                        break;
                    671: 
                    672:                pk_resetcause (pkp, xp);
                    673:                lcp -> lcd_window_condition = lcp -> lcd_rnr_condition =
                    674:                        lcp -> lcd_intrconf_pending = FALSE;
                    675:                lcp -> lcd_output_window = lcp -> lcd_input_window =
                    676:                        lcp -> lcd_last_transmitted_pr = 0;
                    677:                lcp -> lcd_ssn = 0;
                    678:                lcp -> lcd_rsn = MODULUS - 1;
                    679: 
                    680:                lcp -> lcd_template = pk_template (lcp -> lcd_lcn, X25_RESET_CONFIRM);
                    681:                pk_output (lcp);
                    682: 
                    683:                pk_flush (lcp);
                    684:                if (so == 0)
                    685:                        break;
                    686:                wakeup ((caddr_t) & so -> so_timeo);
                    687:                sorwakeup (so);
                    688:                sowwakeup (so);
                    689:                break;
                    690: 
                    691:        /* 
                    692:         *  Reset confirmation received.
                    693:         */
                    694:        case RESET_CONF + DATA_TRANSFER: 
                    695:                if (lcp -> lcd_reset_condition) {
                    696:                        lcp -> lcd_reset_condition = FALSE;
                    697:                        pk_output (lcp);
                    698:                }
                    699:                else
                    700:                        pk_procerror (RESET, lcp, "unexpected packet", 32);
                    701:                break;
                    702: 
                    703:        case DATA + SENT_CLEAR: 
                    704:                ptype = DELETE_PACKET;
                    705:        case RR + SENT_CLEAR: 
                    706:        case RNR + SENT_CLEAR: 
                    707:        case INTERRUPT + SENT_CLEAR: 
                    708:        case INTERRUPT_CONF + SENT_CLEAR: 
                    709:        case RESET + SENT_CLEAR: 
                    710:        case RESET_CONF + SENT_CLEAR: 
                    711:                /* Just ignore p if we have sent a CLEAR already.
                    712:                   */
                    713:                break;
                    714: 
                    715:        /* 
                    716:         *  Restart sets all the permanent virtual circuits to the "Data
                    717:         *  Transfer" stae and  all the switched virtual circuits to the
                    718:         *  "Ready" state.
                    719:         */
                    720:        case RESTART + READY: 
                    721:                switch (pkp -> pk_state) {
                    722:                case DTE_SENT_RESTART: 
                    723:                        /* 
                    724:                         * Restart collision.
                    725:                         * If case the restart cause is "DTE originated" we
                    726:                         * have a DTE-DTE situation and are trying to resolve
                    727:                         * who is going to play DTE/DCE [ISO 8208:4.2-4.5]
                    728:                         */
                    729:                        if (RESTART_DTE_ORIGINATED(xp)) {
                    730:                                pk_restart (pkp, X25_RESTART_DTE_ORIGINATED);
                    731:                                pk_message (0, pkp -> pk_xcp,
                    732:                                            "RESTART collision");
                    733:                                if ((pkp -> pk_restartcolls++) > MAXRESTARTCOLLISIONS) {
                    734:                                        pk_message (0, pkp -> pk_xcp,
                    735:                                                    "excessive RESTART collisions");
                    736:                                        pkp -> pk_restartcolls = 0;
                    737:                                }
                    738:                                break;
                    739:                        }
                    740:                        pkp -> pk_state = DTE_READY;
                    741:                        pkp -> pk_dxerole |= DTE_PLAYDTE;
                    742:                        pkp -> pk_dxerole &= ~DTE_PLAYDCE;
                    743:                        pk_message (0, pkp -> pk_xcp,
                    744:                                "Packet level operational");
                    745:                        pk_message (0, pkp -> pk_xcp, 
                    746:                                    "Assuming DTE role");
                    747:                        if (pkp -> pk_dxerole & DTE_CONNECTPENDING)
                    748:                                pk_callcomplete (pkp);
                    749:                        break;
                    750: 
                    751:                default: 
                    752:                        pk_restart (pkp, -1);
                    753:                        pk_restartcause (pkp, xp);
                    754:                        pkp -> pk_chan[0] -> lcd_template = pk_template (0,
                    755:                                X25_RESTART_CONFIRM);
                    756:                        pk_output (pkp -> pk_chan[0]);
                    757:                        pkp -> pk_state = DTE_READY;
                    758:                        pkp -> pk_dxerole |= RESTART_DTE_ORIGINATED(xp) ? DTE_PLAYDCE :
                    759:                                DTE_PLAYDTE;
                    760:                        if (pkp -> pk_dxerole & DTE_PLAYDTE) {
                    761:                                pkp -> pk_dxerole &= ~DTE_PLAYDCE;
                    762:                                pk_message (0, pkp -> pk_xcp, 
                    763:                                            "Assuming DTE role");
                    764:                        } else {
                    765:                                pkp -> pk_dxerole &= ~DTE_PLAYDTE;
                    766:                                pk_message (0, pkp -> pk_xcp, 
                    767:                                         "Assuming DCE role");
                    768:                        }
                    769:                        if (pkp -> pk_dxerole & DTE_CONNECTPENDING)
                    770:                                pk_callcomplete (pkp);
                    771:                }
                    772:                break;
                    773: 
                    774:        /* 
                    775:         *  Restart confirmation received. All logical channels are set
                    776:         *  to READY. 
                    777:         */
                    778:        case RESTART_CONF + READY: 
                    779:                switch (pkp -> pk_state) {
                    780:                case DTE_SENT_RESTART: 
                    781:                        pkp -> pk_state = DTE_READY;
                    782:                        pkp -> pk_dxerole |= DTE_PLAYDTE;
                    783:                        pkp -> pk_dxerole &= ~DTE_PLAYDCE;
                    784:                        pk_message (0, pkp -> pk_xcp,
                    785:                                    "Packet level operational");
                    786:                        pk_message (0, pkp -> pk_xcp,
                    787:                                    "Assuming DTE role");
                    788:                        if (pkp -> pk_dxerole & DTE_CONNECTPENDING)
                    789:                                pk_callcomplete (pkp);
                    790:                        break;
                    791: 
                    792:                default: 
                    793:                        /* Restart local procedure error. */
                    794:                        pk_restart (pkp, X25_RESTART_LOCAL_PROCEDURE_ERROR);
                    795:                        pkp -> pk_state = DTE_SENT_RESTART;
                    796:                        pkp -> pk_dxerole &= ~(DTE_PLAYDTE | DTE_PLAYDCE);
                    797:                }
                    798:                break;
                    799: 
                    800:        default: 
                    801:                if (lcp) {
                    802:                        pk_procerror (CLEAR, lcp, "unknown packet error", 33);
                    803:                        pk_message (lcn, pkp -> pk_xcp,
                    804:                                "\"%s\" unexpected in \"%s\" state",
                    805:                                pk_name[ptype/MAXSTATES], pk_state[lcdstate]);
                    806:                } else
                    807:                        pk_message (lcn, pkp -> pk_xcp,
                    808:                                "packet arrived on unassigned lcn");
                    809:                break;
                    810:        }
                    811:        if (so == 0 && lcp && lcp -> lcd_upper && lcdstate == DATA_TRANSFER) {
                    812:                if (ptype != DATA && ptype != INTERRUPT)
                    813:                        MCHTYPE(m, MT_CONTROL);
                    814:                lcp -> lcd_upper (lcp, m);
                    815:        } else if (ptype != DATA && ptype != INTERRUPT)
                    816:                m_freem (m);
                    817: }
                    818: 
                    819: static
                    820: prune_dnic (from, to, dnicname, xcp)
                    821: char *from, *to, *dnicname;
                    822: register struct x25config *xcp;
                    823: {
                    824:        register char *cp1 = from, *cp2 = from;
                    825:        if (xcp -> xc_prepnd0 && *cp1 == '0') {
                    826:                from = ++cp1;
                    827:                goto copyrest;
                    828:        }
                    829:        if (xcp -> xc_nodnic) {
                    830:                for (cp1 = dnicname; *cp2 = *cp1++;)
                    831:                        cp2++;
                    832:                cp1 = from;
                    833:        }
                    834: copyrest:
                    835:        for (cp1 = dnicname; *cp2 = *cp1++;)
                    836:                cp2++;
                    837: }
                    838: /* static */
                    839: pk_simple_bsd (from, to, lower, len)
                    840: register octet *from, *to;
                    841: register len, lower;
                    842: {
                    843:        register int c;
                    844:        while (--len >= 0) {
                    845:                c = *from;
                    846:                if (lower & 0x01)
                    847:                        *from++;
                    848:                else
                    849:                        c >>= 4;
                    850:                c &= 0x0f; c |= 0x30; *to++ = c; lower++;
                    851:        }
                    852:        *to = 0;
                    853: }
                    854: 
                    855: /*static octet * */
                    856: pk_from_bcd (a, iscalling, sa, xcp)
                    857: register struct x25_calladdr *a;
                    858: int    iscalling;
                    859: register struct sockaddr_x25 *sa;
                    860: register struct x25config *xcp;
                    861: {
                    862:        octet buf[MAXADDRLN+1];
                    863:        octet *cp;
                    864:        unsigned count;
                    865: 
                    866:        bzero ((caddr_t) sa, sizeof (*sa));
                    867:        sa -> x25_len = sizeof (*sa);
                    868:        sa -> x25_family = AF_CCITT;
                    869:        if (iscalling) {
                    870:                cp = a -> address_field + (X25GBITS(a -> addrlens, called_addrlen) / 2);
                    871:                count = X25GBITS(a -> addrlens, calling_addrlen);
                    872:                pk_simple_bsd (cp, buf, X25GBITS(a -> addrlens, called_addrlen), count);
                    873:        } else {
                    874:                count = X25GBITS(a -> addrlens, called_addrlen);
                    875:                pk_simple_bsd (a -> address_field, buf, 0, count);
                    876:        }
                    877:        if (xcp -> xc_addr.x25_net && (xcp -> xc_nodnic || xcp -> xc_prepnd0)) {
                    878:                octet dnicname[sizeof (long) * NBBY/3 + 2];
                    879: 
                    880:                sprintf ((char *) dnicname, "%d", xcp -> xc_addr.x25_net);
                    881:                prune_dnic ((char *) buf, sa -> x25_addr, dnicname, xcp);
                    882:        } else
                    883:                bcopy ((caddr_t) buf, (caddr_t) sa -> x25_addr, count + 1);
                    884: }
                    885: 
                    886: static
                    887: save_extra (m0, fp, so)
                    888: struct mbuf *m0;
                    889: octet *fp;
                    890: struct socket *so;
                    891: {
                    892:        register struct mbuf *m;
                    893:        struct cmsghdr cmsghdr;
                    894:        if (m = m_copy (m, 0, (int)M_COPYALL)) {
                    895:                int off = fp - mtod (m0, octet *);
                    896:                int len = m -> m_pkthdr.len - off + sizeof (cmsghdr);
                    897:                cmsghdr.cmsg_len = len;
                    898:                cmsghdr.cmsg_level = AF_CCITT;
                    899:                cmsghdr.cmsg_type = PK_FACILITIES;
                    900:                m_adj (m, off);
                    901:                M_PREPEND (m, sizeof (cmsghdr), M_DONTWAIT);
                    902:                if (m == 0)
                    903:                        return;
                    904:                bcopy ((caddr_t)&cmsghdr, mtod (m, caddr_t), sizeof (cmsghdr));
                    905:                MCHTYPE(m, MT_CONTROL);
                    906:                sbappendrecord (&so -> so_rcv, m);
                    907:        }
                    908: }
                    909: 
                    910: /* 
                    911:  * This routine handles incoming call packets. It matches the protocol
                    912:  * field on the Call User Data field (usually the first four bytes) with 
                    913:  * sockets awaiting connections.
                    914:  */
                    915: 
                    916: pk_incoming_call (pkp, m0)
                    917: struct mbuf *m0;
                    918: struct pkcb *pkp;
                    919: {
                    920:        register struct pklcd *lcp = 0, *l;
                    921:        register struct sockaddr_x25 *sa;
                    922:        register struct x25_calladdr *a;
                    923:        register struct socket *so = 0;
                    924:        struct  x25_packet *xp = mtod (m0, struct x25_packet *);
                    925:        struct  mbuf *m;
                    926:        struct  x25config *xcp = pkp -> pk_xcp;
                    927:        int len = m0 -> m_pkthdr.len;
                    928:        int udlen;
                    929:        char *errstr = "server unavailable";
                    930:        octet *u, *facp;
                    931:        int lcn = LCN(xp);
                    932: 
                    933:        /* First, copy the data from the incoming call packet to a X25 address
                    934:           descriptor. It is to be regretted that you have
                    935:           to parse the facilities into a sockaddr to determine
                    936:           if reverse charging is being requested */
                    937:        if ((m = m_get (M_DONTWAIT, MT_SONAME)) == 0)
                    938:                return;
                    939:        sa = mtod (m, struct sockaddr_x25 *);
                    940:        a = (struct x25_calladdr *) &xp -> packet_data;
                    941:        facp = u = (octet *) (a -> address_field +
                    942:                ((X25GBITS(a -> addrlens, called_addrlen) + X25GBITS(a -> addrlens, calling_addrlen) + 1) / 2));
                    943:        u += *u + 1;
                    944:        udlen = min (16, ((octet *) xp) + len - u);
                    945:        if (udlen < 0)
                    946:                udlen = 0;
                    947:        pk_from_bcd (a, 1, sa, pkp -> pk_xcp); /* get calling address */
                    948:        pk_parse_facilities (facp, sa);
                    949:        bcopy ((caddr_t) u, sa -> x25_udata, udlen);
                    950:        sa -> x25_udlen = udlen;
                    951: 
                    952:        /*
                    953:         * Now, loop through the listen sockets looking for a match on the
                    954:         * PID. That is the first few octets of the user data field.
                    955:         * This is the closest thing to a port number for X.25 packets.
                    956:         * It does provide a way of multiplexing services at the user level. 
                    957:         */
                    958: 
                    959:        for (l = pk_listenhead; l; l = l -> lcd_listen) {
                    960:                struct sockaddr_x25 *sxp = l -> lcd_ceaddr;
                    961: 
                    962:                if (bcmp (sxp -> x25_udata, u, sxp -> x25_udlen))
                    963:                        continue;
                    964:                if (sxp -> x25_net &&
                    965:                    sxp -> x25_net != xcp -> xc_addr.x25_net)
                    966:                        continue;
                    967:                /*
                    968:                 * don't accept incoming calls with the D-Bit on
                    969:                 * unless the server agrees
                    970:                 */
                    971:                if (X25GBITS(xp -> bits, d_bit) && !(sxp -> x25_opts.op_flags & X25_DBIT)) {
                    972:                        errstr = "incoming D-Bit mismatch";
                    973:                        break;
                    974:                }
                    975:                /*
                    976:                 * don't accept incoming collect calls unless
                    977:                 * the server sets the reverse charging option.
                    978:                 */
                    979:                if ((sxp -> x25_opts.op_flags & (X25_OLDSOCKADDR|X25_REVERSE_CHARGE)) == 0 &&
                    980:                        sa -> x25_opts.op_flags & X25_REVERSE_CHARGE) {
                    981:                        errstr = "incoming collect call refused";
                    982:                        break;
                    983:                }
                    984:                if (l -> lcd_so) {
                    985:                        if (so = sonewconn (l -> lcd_so, SS_ISCONNECTED))
                    986:                                    lcp = (struct pklcd *) so -> so_pcb;
                    987:                } else 
                    988:                        lcp = pk_attach ((struct socket *) 0);
                    989:                if (lcp == 0) {
                    990:                        /*
                    991:                         * Insufficient space or too many unaccepted
                    992:                         * connections.  Just throw the call away.
                    993:                         */
                    994:                        errstr = "server malfunction";
                    995:                        break;
                    996:                }
                    997:                lcp -> lcd_upper = l -> lcd_upper;
                    998:                lcp -> lcd_upnext = l -> lcd_upnext;
                    999:                lcp -> lcd_lcn = lcn;
                   1000:                lcp -> lcd_state = RECEIVED_CALL;
                   1001:                sa -> x25_opts.op_flags |= (sxp -> x25_opts.op_flags &
                   1002:                        ~X25_REVERSE_CHARGE) | l -> lcd_flags;
                   1003:                pk_assoc (pkp, lcp, sa);
                   1004:                lcp -> lcd_faddr = *sa;
                   1005:                lcp -> lcd_laddr.x25_udlen = sxp -> x25_udlen;
                   1006:                lcp -> lcd_craddr = &lcp -> lcd_faddr;
                   1007:                lcp -> lcd_template = pk_template (lcp -> lcd_lcn, X25_CALL_ACCEPTED);
                   1008:                if (lcp -> lcd_flags & X25_DBIT) {
                   1009:                        if (X25GBITS(xp -> bits, d_bit))
                   1010:                                X25SBITS(mtod (lcp -> lcd_template,
                   1011:                                        struct x25_packet *) -> bits, d_bit, 1);
                   1012:                        else
                   1013:                                lcp -> lcd_flags &= ~X25_DBIT;
                   1014:                }
                   1015:                if (so) {
                   1016:                        pk_output (lcp);
                   1017:                        soisconnected (so);
                   1018:                        if (so -> so_options & SO_OOBINLINE)
                   1019:                                save_extra (m0, facp, so);
                   1020:                } else if (lcp -> lcd_upper) {
                   1021:                        (*lcp -> lcd_upper) (lcp, m0);
                   1022:                }
                   1023:                (void) m_free (m);
                   1024:                return;
                   1025:        }
                   1026: 
                   1027:        /*
                   1028:         * If the call fails for whatever reason, we still need to build a
                   1029:         * skeleton LCD in order to be able to properly  receive the CLEAR
                   1030:         * CONFIRMATION.
                   1031:         */
                   1032: #ifdef WATERLOO                /* be explicit */
                   1033:        if (l == 0 && bcmp (sa -> x25_udata, "ean", 3) == 0)
                   1034:                pk_message (lcn, pkp -> pk_xcp, "host=%s ean%c: %s",
                   1035:                        sa -> x25_addr, sa -> x25_udata[3] & 0xff, errstr);
                   1036:        else if (l == 0 && bcmp (sa -> x25_udata, "\1\0\0\0", 4) == 0)
                   1037:                pk_message (lcn, pkp -> pk_xcp, "host=%s x29d: %s",
                   1038:                        sa -> x25_addr, errstr);
                   1039:        else
                   1040: #endif
                   1041:        pk_message (lcn, pkp -> pk_xcp, "host=%s pid=%x %x %x %x: %s",
                   1042:                sa -> x25_addr, sa -> x25_udata[0] & 0xff,
                   1043:                sa -> x25_udata[1] & 0xff, sa -> x25_udata[2] & 0xff,
                   1044:                sa -> x25_udata[3] & 0xff, errstr);
                   1045:        if ((lcp = pk_attach ((struct socket *)0)) == 0) {
                   1046:                (void) m_free (m);
                   1047:                return;
                   1048:        }
                   1049:        lcp -> lcd_lcn = lcn;
                   1050:        lcp -> lcd_state = RECEIVED_CALL;
                   1051:        pk_assoc (pkp, lcp, sa);
                   1052:        (void) m_free (m);
                   1053:        pk_clear (lcp, 0, 1);
                   1054: }
                   1055: 
                   1056: pk_call_accepted (lcp, m)
                   1057: struct pklcd *lcp;
                   1058: struct mbuf *m;
                   1059: {
                   1060:        register struct x25_calladdr *ap;
                   1061:        register octet *fcp;
                   1062:        struct x25_packet *xp = mtod (m, struct x25_packet *);
                   1063:        int len = m -> m_len;
                   1064: 
                   1065:        lcp -> lcd_state = DATA_TRANSFER;
                   1066:        if (lcp -> lcd_so)
                   1067:                soisconnected (lcp -> lcd_so);
                   1068:        if ((lcp -> lcd_flags & X25_DBIT) && (X25GBITS(xp -> bits, d_bit) == 0))
                   1069:                lcp -> lcd_flags &= ~X25_DBIT;
                   1070:        if (len > 3) {
                   1071:                ap = (struct x25_calladdr *) &xp -> packet_data;
                   1072:                fcp = (octet *) ap -> address_field + (X25GBITS(ap -> addrlens, calling_addrlen) +
                   1073:                        X25GBITS(ap -> addrlens, called_addrlen) + 1) / 2;
                   1074:                if (fcp + *fcp <= ((octet *) xp) + len)
                   1075:                        pk_parse_facilities (fcp, lcp -> lcd_ceaddr);
                   1076:        }
                   1077:        pk_assoc (lcp -> lcd_pkp, lcp, lcp -> lcd_ceaddr);
                   1078:        if (lcp -> lcd_so == 0 && lcp -> lcd_upper)
                   1079:                lcp -> lcd_upper (lcp, m);
                   1080: }
                   1081: 
                   1082: pk_parse_facilities (fcp, sa)
                   1083: register octet *fcp;
                   1084: register struct sockaddr_x25 *sa;
                   1085: {
                   1086:        register octet *maxfcp;
                   1087: 
                   1088:        maxfcp = fcp + *fcp;
                   1089:        fcp++;
                   1090:        while (fcp < maxfcp) {
                   1091:                /*
                   1092:                 * Ignore national DCE or DTE facilities
                   1093:                 */
                   1094:                if (*fcp == 0 || *fcp == 0xff)
                   1095:                        break;
                   1096:                switch (*fcp) {
                   1097:                case FACILITIES_WINDOWSIZE:
                   1098:                        sa -> x25_opts.op_wsize = fcp[1];
                   1099:                        fcp += 3;
                   1100:                        break;
                   1101: 
                   1102:                case FACILITIES_PACKETSIZE:
                   1103:                        sa -> x25_opts.op_psize = fcp[1];
                   1104:                        fcp += 3;
                   1105:                        break;
                   1106: 
                   1107:                case FACILITIES_THROUGHPUT:
                   1108:                        sa -> x25_opts.op_speed = fcp[1];
                   1109:                        fcp += 2;
                   1110:                        break;
                   1111: 
                   1112:                case FACILITIES_REVERSE_CHARGE:
                   1113:                        if (fcp[1] & 01)
                   1114:                                sa -> x25_opts.op_flags |= X25_REVERSE_CHARGE;
                   1115:                        /*
                   1116:                         * Datapac specific: for a X.25(1976) DTE, bit 2
                   1117:                         * indicates a "hi priority" (eg. international) call.
                   1118:                         */
                   1119:                        if (fcp[1] & 02 && sa -> x25_opts.op_psize == 0)
                   1120:                                sa -> x25_opts.op_psize = X25_PS128;
                   1121:                        fcp += 2;
                   1122:                        break;
                   1123: 
                   1124:                default:
                   1125: /*printf("unknown facility %x, class=%d\n", *fcp, (*fcp & 0xc0) >> 6);*/
                   1126:                        switch ((*fcp & 0xc0) >> 6) {
                   1127:                        case 0:                 /* class A */
                   1128:                                fcp += 2;
                   1129:                                break;
                   1130: 
                   1131:                        case 1:
                   1132:                                fcp += 3;
                   1133:                                break;
                   1134: 
                   1135:                        case 2:
                   1136:                                fcp += 4;
                   1137:                                break;
                   1138: 
                   1139:                        case 3:
                   1140:                                fcp++;
                   1141:                                fcp += *fcp;
                   1142:                        }
                   1143:                }
                   1144:        }
                   1145: }

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.