Annotation of XNU/bsd/netiso/tp_usrreq.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) 1991, 1993
                     24:  *     The Regents of the University of California.  All rights reserved.
                     25:  *
                     26:  * Redistribution and use in source and binary forms, with or without
                     27:  * modification, are permitted provided that the following conditions
                     28:  * are met:
                     29:  * 1. Redistributions of source code must retain the above copyright
                     30:  *    notice, this list of conditions and the following disclaimer.
                     31:  * 2. Redistributions in binary form must reproduce the above copyright
                     32:  *    notice, this list of conditions and the following disclaimer in the
                     33:  *    documentation and/or other materials provided with the distribution.
                     34:  * 3. All advertising materials mentioning features or use of this software
                     35:  *    must display the following acknowledgement:
                     36:  *     This product includes software developed by the University of
                     37:  *     California, Berkeley and its contributors.
                     38:  * 4. Neither the name of the University nor the names of its contributors
                     39:  *    may be used to endorse or promote products derived from this software
                     40:  *    without specific prior written permission.
                     41:  *
                     42:  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
                     43:  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
                     44:  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
                     45:  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
                     46:  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
                     47:  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
                     48:  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
                     49:  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
                     50:  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
                     51:  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
                     52:  * SUCH DAMAGE.
                     53:  *
                     54:  *     @(#)tp_usrreq.c 8.1 (Berkeley) 6/10/93
                     55:  */
                     56: 
                     57: /***********************************************************
                     58:                                Copyright IBM Corporation 1987
                     59: 
                     60:                       All Rights Reserved
                     61: 
                     62: Permission to use, copy, modify, and distribute this software and its 
                     63: documentation for any purpose and without fee is hereby granted, 
                     64: provided that the above copyright notice appear in all copies and that
                     65: both that copyright notice and this permission notice appear in 
                     66: supporting documentation, and that the name of IBM not be
                     67: used in advertising or publicity pertaining to distribution of the
                     68: software without specific, written prior permission.  
                     69: 
                     70: IBM DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
                     71: ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
                     72: IBM BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
                     73: ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
                     74: WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
                     75: ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
                     76: SOFTWARE.
                     77: 
                     78: ******************************************************************/
                     79: 
                     80: /*
                     81:  * ARGO Project, Computer Sciences Dept., University of Wisconsin - Madison
                     82:  */
                     83: /* 
                     84:  * ARGO TP
                     85:  *
                     86:  * tp_usrreq(), the fellow that gets called from most of the socket code.
                     87:  * Pretty straighforward.
                     88:  * THe only really awful stuff here is the OOB processing, which is done
                     89:  * wholly here.
                     90:  * tp_rcvoob() and tp_sendoob() are contained here and called by tp_usrreq().
                     91:  */
                     92: 
                     93: #include <sys/param.h>
                     94: #include <sys/systm.h>
                     95: #include <sys/mbuf.h>
                     96: #include <sys/socket.h>
                     97: #include <sys/socketvar.h>
                     98: #include <sys/domain.h>
                     99: #include <sys/protosw.h>
                    100: #include <sys/errno.h>
                    101: #include <sys/time.h>
                    102: 
                    103: #include <netiso/tp_param.h>
                    104: #include <netiso/tp_timer.h>
                    105: #include <netiso/tp_stat.h>
                    106: #include <netiso/tp_seq.h>
                    107: #include <netiso/tp_ip.h>
                    108: #include <netiso/tp_pcb.h>
                    109: #include <netiso/argo_debug.h>
                    110: #include <netiso/tp_trace.h>
                    111: #include <netiso/tp_meas.h>
                    112: #include <netiso/iso.h>
                    113: #include <netiso/iso_errno.h>
                    114: 
                    115: int tp_attach(), tp_driver(), tp_pcbbind();
                    116: int TNew;
                    117: int TPNagle1, TPNagle2;
                    118: struct tp_pcb *tp_listeners, *tp_intercepts;
                    119: 
                    120: #ifdef ARGO_DEBUG
                    121: /*
                    122:  * CALLED FROM:
                    123:  *  anywhere you want to debug...
                    124:  * FUNCTION and ARGUMENTS:
                    125:  *  print (str) followed by the control info in the mbufs of an mbuf chain (n)
                    126:  */
                    127: void
                    128: dump_mbuf(n, str)
                    129:        struct mbuf *n;
                    130:        char *str;
                    131: {
                    132:        struct mbuf *nextrecord;
                    133: 
                    134:        printf("dump %s\n", str);
                    135: 
                    136:        if (n == MNULL)  {
                    137:                printf("EMPTY:\n");
                    138:                return;
                    139:        }
                    140:                
                    141:        while (n) {
                    142:                nextrecord = n->m_act;
                    143:                printf("RECORD:\n");
                    144:                while (n) {
                    145:                        printf("%x : Len %x Data %x A %x Nx %x Tp %x\n",
                    146:                                n, n->m_len, n->m_data, n->m_act, n->m_next, n->m_type);
                    147: #ifdef notdef
                    148:                        {
                    149:                                register char *p = mtod(n, char *);
                    150:                                register int i;
                    151: 
                    152:                                printf("data: ");
                    153:                                for (i = 0; i < n->m_len; i++) {
                    154:                                        if (i%8 == 0)
                    155:                                                printf("\n");
                    156:                                        printf("0x%x ", *(p+i));
                    157:                                }
                    158:                                printf("\n");
                    159:                        }
                    160: #endif /* notdef */
                    161:                        if (n->m_next == n) {
                    162:                                printf("LOOP!\n");
                    163:                                return;
                    164:                        }
                    165:                        n = n->m_next;
                    166:                }
                    167:                n = nextrecord;
                    168:        }
                    169:        printf("\n");
                    170: }
                    171: 
                    172: #endif /* ARGO_DEBUG */
                    173: 
                    174: /*
                    175:  * CALLED FROM:
                    176:  *  tp_usrreq(), PRU_RCVOOB
                    177:  * FUNCTION and ARGUMENTS:
                    178:  *     Copy data from the expedited data socket buffer into
                    179:  *     the pre-allocated mbuf m.
                    180:  *     There is an isomorphism between XPD TPDUs and expedited data TSDUs.
                    181:  *     XPD tpdus are limited to 16 bytes of data so they fit in one mbuf.
                    182:  * RETURN VALUE:
                    183:  *  EINVAL if debugging is on and a disaster has occurred
                    184:  *  ENOTCONN if the socket isn't connected
                    185:  *  EWOULDBLOCK if the socket is in non-blocking mode and there's no
                    186:  *             xpd data in the buffer
                    187:  *  E* whatever is returned from the fsm.
                    188:  */
                    189: tp_rcvoob(tpcb, so, m, outflags, inflags)
                    190:        struct tp_pcb   *tpcb;
                    191:        register struct socket  *so;
                    192:        register struct mbuf    *m;
                    193:        int                     *outflags;
                    194:        int                     inflags;
                    195: {
                    196:        register struct mbuf *n;
                    197:        register struct sockbuf *sb = &so->so_rcv;
                    198:        struct tp_event E;
                    199:        int error = 0;
                    200:        register struct mbuf **nn;
                    201: 
                    202:        IFDEBUG(D_XPD)
                    203:                printf("PRU_RCVOOB, sostate 0x%x\n", so->so_state);
                    204:        ENDDEBUG
                    205: 
                    206:        /* if you use soreceive */
                    207:        if (m == MNULL)
                    208:                return ENOBUFS;
                    209: 
                    210: restart:
                    211:        if ((((so->so_state & SS_ISCONNECTED) == 0)
                    212:                 || (so->so_state & SS_ISDISCONNECTING) != 0) &&
                    213:                (so->so_proto->pr_flags & PR_CONNREQUIRED)) {
                    214:                        return ENOTCONN;
                    215:        }
                    216: 
                    217:        /* Take the first mbuf off the chain.
                    218:         * Each XPD TPDU gives you a complete TSDU so the chains don't get 
                    219:         * coalesced, but one TSDU may span several mbufs.
                    220:         * Nevertheless, since n should have a most 16 bytes, it
                    221:         * will fit into m.  (size was checked in tp_input() )
                    222:         */
                    223: 
                    224:        /*
                    225:         * Code for excision of OOB data should be added to
                    226:         * uipc_socket2.c (like sbappend).
                    227:         */
                    228:        
                    229:        sblock(sb, M_WAIT);
                    230:        for (nn = &sb->sb_mb; n = *nn; nn = &n->m_act)
                    231:                if (n->m_type == MT_OOBDATA)
                    232:                        break;
                    233: 
                    234:        if (n == 0) {
                    235:                IFDEBUG(D_XPD)
                    236:                        printf("RCVOOB: empty queue!\n");
                    237:                ENDDEBUG
                    238:                sbunlock(sb);
                    239:                if (so->so_state & SS_NBIO) {
                    240:                        return  EWOULDBLOCK;
                    241:                }
                    242:                sbwait(sb);
                    243:                goto restart;
                    244:        }
                    245:        m->m_len = 0;
                    246: 
                    247:        /* Assuming at most one xpd tpdu is in the buffer at once */
                    248:        while (n != MNULL) {
                    249:                m->m_len += n->m_len;
                    250:                bcopy(mtod(n, caddr_t), mtod(m, caddr_t), (unsigned)n->m_len);
                    251:                m->m_data += n->m_len; /* so mtod() in bcopy() above gives right addr */
                    252:                n = n->m_next;
                    253:        }
                    254:        m->m_data = m->m_dat;
                    255:        m->m_flags |= M_EOR;
                    256: 
                    257:        IFDEBUG(D_XPD)
                    258:                printf("tp_rcvoob: xpdlen 0x%x\n", m->m_len);
                    259:                dump_mbuf(so->so_rcv.sb_mb, "RCVOOB: Rcv socketbuf");
                    260:                dump_mbuf(sb->sb_mb, "RCVOOB: Xrcv socketbuf");
                    261:        ENDDEBUG
                    262: 
                    263:        if ((inflags & MSG_PEEK) == 0) {
                    264:                n = *nn;
                    265:                *nn = n->m_act;
                    266:                for (; n; n = m_free(n)) 
                    267:                        sbfree(sb, n);
                    268:        }
                    269: 
                    270: release:
                    271:        sbunlock(sb);
                    272: 
                    273:        IFTRACE(D_XPD)
                    274:                tptraceTPCB(TPPTmisc, "PRU_RCVOOB @ release sb_cc m_len",
                    275:                        tpcb->tp_Xrcv.sb_cc, m->m_len, 0, 0);
                    276:        ENDTRACE
                    277:        if (error == 0)
                    278:                error = DoEvent(T_USR_Xrcvd); 
                    279:        return error;
                    280: }
                    281: 
                    282: /*
                    283:  * CALLED FROM:
                    284:  *  tp_usrreq(), PRU_SENDOOB
                    285:  * FUNCTION and ARGUMENTS:
                    286:  *     Send what's in the mbuf chain (m) as an XPD TPDU.
                    287:  *     The mbuf may not contain more then 16 bytes of data.
                    288:  *     XPD TSDUs aren't segmented, so they translate into
                    289:  *     exactly one XPD TPDU, with EOT bit set.
                    290:  * RETURN VALUE:
                    291:  *  EWOULDBLOCK if socket is in non-blocking mode and the previous
                    292:  *   xpd data haven't been acked yet.
                    293:  *  EMSGSIZE if trying to send > max-xpd bytes (16)
                    294:  *  ENOBUFS if ran out of mbufs
                    295:  */
                    296: tp_sendoob(tpcb, so, xdata, outflags)
                    297:        struct tp_pcb   *tpcb;
                    298:        register struct socket  *so;
                    299:        register struct mbuf    *xdata;
                    300:        int                     *outflags; /* not used */
                    301: {
                    302:        /*
                    303:         * Each mbuf chain represents a sequence # in the XPD seq space.
                    304:         * The first one in the queue has sequence # tp_Xuna.
                    305:         * When we add to the XPD queue, we stuff a zero-length
                    306:         * mbuf (mark) into the DATA queue, with its sequence number in m_next
                    307:         * to be assigned to this XPD tpdu, so data xfer can stop
                    308:         * when it reaches the zero-length mbuf if this XPD TPDU hasn't
                    309:         * yet been acknowledged.  
                    310:         */
                    311:        register struct sockbuf *sb = &(tpcb->tp_Xsnd);
                    312:        register struct mbuf    *xmark;
                    313:        register int                    len=0;
                    314:        struct tp_event E;
                    315: 
                    316:        IFDEBUG(D_XPD)
                    317:                printf("tp_sendoob:");
                    318:                if (xdata)
                    319:                        printf("xdata len 0x%x\n", xdata->m_len);
                    320:        ENDDEBUG
                    321:        /* DO NOT LOCK the Xsnd buffer!!!! You can have at MOST one 
                    322:         * socket buf locked at any time!!! (otherwise you might
                    323:         * sleep() in sblock() w/ a signal pending and cause the
                    324:         * system call to be aborted w/ a locked socketbuf, which
                    325:         * is a problem.  So the so_snd buffer lock
                    326:         * (done in sosend()) serves as the lock for Xpd.
                    327:         */
                    328:        if (sb->sb_mb) { /* Anything already in eXpedited data sockbuf? */
                    329:                if (so->so_state & SS_NBIO) {
                    330:                        return EWOULDBLOCK;
                    331:                }
                    332:                while (sb->sb_mb) {
                    333:                        sbunlock(&so->so_snd); /* already locked by sosend */
                    334:                        sbwait(&so->so_snd);
                    335:                        sblock(&so->so_snd, M_WAIT);  /* sosend will unlock on return */
                    336:                }
                    337:        }
                    338: 
                    339:        if (xdata == (struct mbuf *)0) {
                    340:                /* empty xpd packet */
                    341:                MGETHDR(xdata, M_WAIT, MT_OOBDATA);
                    342:                if (xdata == NULL) {
                    343:                        return ENOBUFS;
                    344:                }
                    345:                xdata->m_len = 0;
                    346:                xdata->m_pkthdr.len = 0;
                    347:        }
                    348:        IFDEBUG(D_XPD)
                    349:                printf("tp_sendoob 1:");
                    350:                if (xdata)
                    351:                        printf("xdata len 0x%x\n", xdata->m_len);
                    352:        ENDDEBUG
                    353:        xmark = xdata; /* temporary use of variable xmark */
                    354:        while (xmark) {
                    355:                len += xmark->m_len;
                    356:                xmark = xmark->m_next;
                    357:        }
                    358:        if (len > TP_MAX_XPD_DATA) {
                    359:                return EMSGSIZE;
                    360:        }
                    361:        IFDEBUG(D_XPD)
                    362:                printf("tp_sendoob 2:");
                    363:                if (xdata)
                    364:                        printf("xdata len 0x%x\n", len);
                    365:        ENDDEBUG
                    366: 
                    367: 
                    368:        IFTRACE(D_XPD)
                    369:                tptraceTPCB(TPPTmisc, "XPD mark m_next ", xdata->m_next, 0, 0, 0);
                    370:        ENDTRACE
                    371: 
                    372:        sbappendrecord(sb, xdata);      
                    373: 
                    374:        IFDEBUG(D_XPD)
                    375:                printf("tp_sendoob len 0x%x\n", len);
                    376:                dump_mbuf(so->so_snd.sb_mb, "XPD request Regular sndbuf:");
                    377:                dump_mbuf(tpcb->tp_Xsnd.sb_mb, "XPD request Xsndbuf:");
                    378:        ENDDEBUG
                    379:        return DoEvent(T_XPD_req); 
                    380: }
                    381: 
                    382: /*
                    383:  * CALLED FROM:
                    384:  *  the socket routines
                    385:  * FUNCTION and ARGUMENTS:
                    386:  *     Handles all "user requests" except the [gs]ockopts() requests.
                    387:  *     The argument (req) is the request type (PRU*), 
                    388:  *     (m) is an mbuf chain, generally used for send and
                    389:  *     receive type requests only.
                    390:  *     (nam) is used for addresses usually, in particular for the bind request.
                    391:  * 
                    392:  */
                    393: /*ARGSUSED*/
                    394: ProtoHook
                    395: tp_usrreq(so, req, m, nam, controlp)
                    396:        struct socket *so;
                    397:        u_int req;
                    398:        struct mbuf *m, *nam, *controlp;
                    399: {      
                    400:        register struct tp_pcb *tpcb =  sototpcb(so);
                    401:        int s = splnet();
                    402:        int error = 0;
                    403:        int flags, *outflags = &flags; 
                    404:        u_long eotsdu = 0;
                    405:        struct tp_event E;
                    406: 
                    407:        IFDEBUG(D_REQUEST)
                    408:                printf("usrreq(0x%x,%d,0x%x,0x%x,0x%x)\n",so,req,m,nam,outflags);
                    409:                if (so->so_error)
                    410:                        printf("WARNING!!! so->so_error is 0x%x\n", so->so_error);
                    411:        ENDDEBUG
                    412:        IFTRACE(D_REQUEST)
                    413:                tptraceTPCB(TPPTusrreq, "req so m state [", req, so, m, 
                    414:                        tpcb?tpcb->tp_state:0);
                    415:        ENDTRACE
                    416: 
                    417:        if ((u_int)tpcb == 0 && req != PRU_ATTACH) {
                    418:                IFTRACE(D_REQUEST)
                    419:                        tptraceTPCB(TPPTusrreq, "req failed NO TPCB[", 0, 0, 0, 0);
                    420:                ENDTRACE
                    421:                splx(s);
                    422:                return ENOTCONN;
                    423:        }
                    424: 
                    425:        switch (req) {
                    426: 
                    427:        case PRU_ATTACH:
                    428:                if (tpcb) {
                    429:                        error = EISCONN;
                    430:                } else if ((error = tp_attach(so, (int)nam)) == 0)
                    431:                        tpcb = sototpcb(so);
                    432:                break;
                    433: 
                    434:        case PRU_ABORT:         /* called from close() */
                    435:                /* called for each incoming connect queued on the 
                    436:                 *      parent (accepting) socket 
                    437:                 */
                    438:                if (tpcb->tp_state == TP_OPEN || tpcb->tp_state == TP_CONFIRMING) {
                    439:                        E.ATTR(T_DISC_req).e_reason = E_TP_NO_SESSION;
                    440:                        error = DoEvent(T_DISC_req); /* pretend it was a close() */
                    441:                        break;
                    442:                } /* else DROP THROUGH */
                    443: 
                    444:        case PRU_DETACH:        /* called from close() */
                    445:                /* called only after disconnect was called */
                    446:                error = DoEvent(T_DETACH);
                    447:                if (tpcb->tp_state == TP_CLOSED) {
                    448:                        if (tpcb->tp_notdetached) {
                    449:                                IFDEBUG(D_CONN)
                    450:                                        printf("PRU_DETACH: not detached\n");
                    451:                                ENDDEBUG
                    452:                                tp_detach(tpcb);
                    453:                        }
                    454:                        FREE((caddr_t)tpcb, M_PCB);
                    455:                        tpcb = 0;
                    456:                }
                    457:                break;
                    458: 
                    459:        case PRU_SHUTDOWN:
                    460:                /* recv end may have been released; local credit might be zero  */
                    461:        case PRU_DISCONNECT:
                    462:                E.ATTR(T_DISC_req).e_reason = E_TP_NORMAL_DISC;
                    463:                error = DoEvent(T_DISC_req);
                    464:                break;
                    465: 
                    466:        case PRU_BIND:
                    467:                error =  tp_pcbbind(tpcb, nam);
                    468:                break;
                    469: 
                    470:        case PRU_LISTEN:
                    471:                if (tpcb->tp_state != TP_CLOSED || tpcb->tp_lsuffixlen == 0 ||
                    472:                                tpcb->tp_next == 0)
                    473:                        error = EINVAL;
                    474:                else {
                    475:                        register struct tp_pcb **tt;
                    476:                        remque(tpcb);
                    477:                        tpcb->tp_next = tpcb->tp_prev = tpcb;
                    478:                        for (tt = &tp_listeners; *tt; tt = &((*tt)->tp_nextlisten))
                    479:                                if ((*tt)->tp_lsuffixlen)
                    480:                                        break;
                    481:                        tpcb->tp_nextlisten = *tt;
                    482:                        *tt = tpcb;
                    483:                        error = DoEvent(T_LISTEN_req);
                    484:                }
                    485:                break;
                    486: 
                    487:        case PRU_CONNECT2:
                    488:                error = EOPNOTSUPP; /* for unix domain sockets */
                    489:                break;
                    490: 
                    491:        case PRU_CONNECT:
                    492:                IFTRACE(D_CONN)
                    493:                        tptraceTPCB(TPPTmisc, 
                    494:                        "PRU_CONNECT: so 0x%x *SHORT_LSUFXP(tpcb) 0x%x lsuflen 0x%x, class 0x%x",
                    495:                        tpcb->tp_sock, *SHORT_LSUFXP(tpcb), tpcb->tp_lsuffixlen,
                    496:                                tpcb->tp_class);
                    497:                ENDTRACE
                    498:                IFDEBUG(D_CONN)
                    499:                        printf("PRU_CONNECT: so *SHORT_LSUFXP(tpcb) 0x%x lsuflen 0x%x, class 0x%x",
                    500:                        tpcb->tp_sock, *SHORT_LSUFXP(tpcb), tpcb->tp_lsuffixlen,
                    501:                                tpcb->tp_class);
                    502:                ENDDEBUG
                    503:                if (tpcb->tp_lsuffixlen == 0) {
                    504:                        if (error = tp_pcbbind(tpcb, MNULL)) {
                    505:                                IFDEBUG(D_CONN)
                    506:                                        printf("pcbbind returns error 0x%x\n", error);
                    507:                                ENDDEBUG
                    508:                                break;
                    509:                        }
                    510:                }
                    511:                IFDEBUG(D_CONN)
                    512:                        printf("isop 0x%x isop->isop_socket offset 12 :\n", tpcb->tp_npcb);
                    513:                        dump_buf(tpcb->tp_npcb, 16);
                    514:                ENDDEBUG
                    515:                if (error = tp_route_to(nam, tpcb, /* channel */0))
                    516:                        break;
                    517:                IFDEBUG(D_CONN)
                    518:                        printf(
                    519:                                "PRU_CONNECT after tpcb 0x%x so 0x%x npcb 0x%x flags 0x%x\n", 
                    520:                                tpcb, so, tpcb->tp_npcb, tpcb->tp_flags);
                    521:                        printf("isop 0x%x isop->isop_socket offset 12 :\n", tpcb->tp_npcb);
                    522:                        dump_buf(tpcb->tp_npcb, 16);
                    523:                ENDDEBUG
                    524:                if (tpcb->tp_fsuffixlen ==  0) {
                    525:                        /* didn't set peer extended suffix */
                    526:                        (tpcb->tp_nlproto->nlp_getsufx)(tpcb->tp_npcb, &tpcb->tp_fsuffixlen,
                    527:                                tpcb->tp_fsuffix, TP_FOREIGN);
                    528:                }
                    529:                if (tpcb->tp_state == TP_CLOSED) {
                    530:                        soisconnecting(so);  
                    531:                        error = DoEvent(T_CONN_req);
                    532:                } else {
                    533:                        (tpcb->tp_nlproto->nlp_pcbdisc)(tpcb->tp_npcb);
                    534:                        error = EISCONN;
                    535:                }
                    536:                IFPERF(tpcb)
                    537:                        u_int lsufx, fsufx;
                    538:                        lsufx = *(u_short *)(tpcb->tp_lsuffix);
                    539:                        fsufx = *(u_short *)(tpcb->tp_fsuffix);
                    540: 
                    541:                        tpmeas(tpcb->tp_lref, 
                    542:                                TPtime_open | (tpcb->tp_xtd_format << 4), 
                    543:                                &time, lsufx, fsufx, tpcb->tp_fref);
                    544:                ENDPERF
                    545:                break;
                    546: 
                    547:        case PRU_ACCEPT: 
                    548:                (tpcb->tp_nlproto->nlp_getnetaddr)(tpcb->tp_npcb, nam, TP_FOREIGN);
                    549:                IFDEBUG(D_REQUEST)
                    550:                        printf("ACCEPT PEERADDDR:");
                    551:                        dump_buf(mtod(nam, char *), nam->m_len);
                    552:                ENDDEBUG
                    553:                IFPERF(tpcb)
                    554:                        u_int lsufx, fsufx;
                    555:                        lsufx = *(u_short *)(tpcb->tp_lsuffix);
                    556:                        fsufx = *(u_short *)(tpcb->tp_fsuffix);
                    557: 
                    558:                        tpmeas(tpcb->tp_lref, TPtime_open, 
                    559:                                &time, lsufx, fsufx, tpcb->tp_fref);
                    560:                ENDPERF
                    561:                break;
                    562: 
                    563:        case PRU_RCVD:
                    564:                if (so->so_state & SS_ISCONFIRMING) {
                    565:                        if (tpcb->tp_state == TP_CONFIRMING)
                    566:                                error = tp_confirm(tpcb);
                    567:                        break;
                    568:                }
                    569:                IFTRACE(D_DATA)
                    570:                        tptraceTPCB(TPPTmisc,
                    571:                        "RCVD BF: lcredit sent_lcdt cc hiwat \n",
                    572:                                tpcb->tp_lcredit, tpcb->tp_sent_lcdt,
                    573:                                so->so_rcv.sb_cc, so->so_rcv.sb_hiwat);
                    574:                        LOCAL_CREDIT(tpcb);
                    575:                        tptraceTPCB(TPPTmisc, 
                    576:                                "PRU_RCVD AF sbspace lcredit hiwat cc",
                    577:                                sbspace(&so->so_rcv), tpcb->tp_lcredit,
                    578:                                so->so_rcv.sb_cc, so->so_rcv.sb_hiwat);
                    579:                ENDTRACE
                    580:                IFDEBUG(D_REQUEST)
                    581:                        printf("RCVD: cc %d space %d hiwat %d\n",
                    582:                                so->so_rcv.sb_cc, sbspace(&so->so_rcv),
                    583:                                so->so_rcv.sb_hiwat);
                    584:                ENDDEBUG
                    585:                if (((int)nam) & MSG_OOB)
                    586:                        error = DoEvent(T_USR_Xrcvd); 
                    587:                else 
                    588:                        error = DoEvent(T_USR_rcvd); 
                    589:                break;
                    590: 
                    591:        case PRU_RCVOOB:
                    592:                if ((so->so_state & SS_ISCONNECTED) == 0) {
                    593:                        error = ENOTCONN;
                    594:                        break;
                    595:                }
                    596:                if (! tpcb->tp_xpd_service) {
                    597:                        error = EOPNOTSUPP;
                    598:                        break;
                    599:                }
                    600:                /* kludge - nam is really flags here */
                    601:                error = tp_rcvoob(tpcb, so, m, outflags, (int)nam);
                    602:                break;
                    603: 
                    604:        case PRU_SEND:
                    605:        case PRU_SENDOOB:
                    606:                if (controlp) {
                    607:                        error = tp_snd_control(controlp, so, &m);
                    608:                        controlp = NULL;
                    609:                        if (error)
                    610:                                break;
                    611:                }
                    612:                if ((so->so_state & SS_ISCONFIRMING) &&
                    613:                    (tpcb->tp_state == TP_CONFIRMING) &&
                    614:                    (error = tp_confirm(tpcb)))
                    615:                            break;
                    616:                if (req == PRU_SENDOOB) {
                    617:                        error = (tpcb->tp_xpd_service == 0) ?
                    618:                                                EOPNOTSUPP : tp_sendoob(tpcb, so, m, outflags);
                    619:                        break;
                    620:                }
                    621:                if (m == 0)
                    622:                        break;
                    623:                if (m->m_flags & M_EOR) {
                    624:                        eotsdu = 1;
                    625:                        m->m_flags &= ~M_EOR;
                    626:                }
                    627:                if (eotsdu == 0 && m->m_pkthdr.len == 0)
                    628:                        break;
                    629:                if (tpcb->tp_state != TP_AKWAIT && tpcb->tp_state != TP_OPEN) {
                    630:                        error = ENOTCONN;
                    631:                        break;
                    632:                }
                    633:                /*
                    634:                 * The protocol machine copies mbuf chains,
                    635:                 * prepends headers, assigns seq numbers, and
                    636:                 * puts the packets on the device.
                    637:                 * When they are acked they are removed from the socket buf.
                    638:                 *
                    639:                 * sosend calls this up until sbspace goes negative.
                    640:                 * Sbspace may be made negative by appending this mbuf chain,
                    641:                 * possibly by a whole cluster.
                    642:                 */
                    643:                {
                    644:                        /*
                    645:                         * Could have eotsdu and no data.(presently MUST have
                    646:                         * an mbuf though, even if its length == 0) 
                    647:                         */
                    648:                        int totlen = m->m_pkthdr.len;
                    649:                        struct sockbuf *sb = &so->so_snd;
                    650:                        IFPERF(tpcb)
                    651:                           PStat(tpcb, Nb_from_sess) += totlen;
                    652:                           tpmeas(tpcb->tp_lref, TPtime_from_session, 0, 0, 
                    653:                                        PStat(tpcb, Nb_from_sess), totlen);
                    654:                        ENDPERF
                    655:                        IFDEBUG(D_SYSCALL)
                    656:                                printf(
                    657:                                "PRU_SEND: eot %d before sbappend 0x%x len 0x%x to sb @ 0x%x\n",
                    658:                                        eotsdu, m, totlen, sb);
                    659:                                dump_mbuf(sb->sb_mb, "so_snd.sb_mb");
                    660:                                dump_mbuf(m, "m : to be added");
                    661:                        ENDDEBUG
                    662:                        tp_packetize(tpcb, m, eotsdu);
                    663:                        IFDEBUG(D_SYSCALL)
                    664:                                printf("PRU_SEND: eot %d after sbappend 0x%x\n", eotsdu, m);
                    665:                                dump_mbuf(sb->sb_mb, "so_snd.sb_mb");
                    666:                        ENDDEBUG
                    667:                        if (tpcb->tp_state == TP_OPEN)
                    668:                                error = DoEvent(T_DATA_req); 
                    669:                        IFDEBUG(D_SYSCALL)
                    670:                                printf("PRU_SEND: after driver error 0x%x \n",error);
                    671:                                printf("so_snd 0x%x cc 0t%d mbcnt 0t%d\n",
                    672:                                                sb, sb->sb_cc, sb->sb_mbcnt);
                    673:                                dump_mbuf(sb->sb_mb, "so_snd.sb_mb after driver");
                    674:                        ENDDEBUG
                    675:                }
                    676:                break;
                    677: 
                    678:        case PRU_SOCKADDR:
                    679:                (tpcb->tp_nlproto->nlp_getnetaddr)(tpcb->tp_npcb, nam, TP_LOCAL);
                    680:                break;
                    681: 
                    682:        case PRU_PEERADDR:
                    683:                (tpcb->tp_nlproto->nlp_getnetaddr)(tpcb->tp_npcb, nam, TP_FOREIGN);
                    684:                break;
                    685: 
                    686:        case PRU_CONTROL:
                    687:                error = EOPNOTSUPP;
                    688:                break;
                    689: 
                    690:        case PRU_PROTOSEND:
                    691:        case PRU_PROTORCV:
                    692:        case PRU_SENSE:
                    693:        case PRU_SLOWTIMO:
                    694:        case PRU_FASTTIMO:
                    695:                error = EOPNOTSUPP;
                    696:                break;
                    697: 
                    698:        default:
                    699: #ifdef ARGO_DEBUG
                    700:                printf("tp_usrreq UNKNOWN PRU %d\n", req);
                    701: #endif /* ARGO_DEBUG */
                    702:                error = EOPNOTSUPP;
                    703:        }
                    704: 
                    705:        IFDEBUG(D_REQUEST)
                    706:                printf("%s, so 0x%x, tpcb 0x%x, error %d, state %d\n",
                    707:                        "returning from tp_usrreq", so, tpcb, error,
                    708:                        tpcb ? tpcb->tp_state : 0);
                    709:        ENDDEBUG
                    710:        IFTRACE(D_REQUEST)
                    711:                tptraceTPCB(TPPTusrreq, "END req so m state [", req, so, m, 
                    712:                        tpcb ? tpcb->tp_state : 0);
                    713:        ENDTRACE
                    714:        if (controlp) {
                    715:                m_freem(controlp);
                    716:                printf("control data unexpectedly retained in tp_usrreq()");
                    717:        }
                    718:        splx(s);
                    719:        return error;
                    720: }
                    721: tp_ltrace(so, uio)
                    722: struct socket *so;
                    723: struct uio *uio;
                    724: {
                    725:        IFTRACE(D_DATA)
                    726:                register struct tp_pcb *tpcb =  sototpcb(so);
                    727:                if (tpcb) {
                    728:                        tptraceTPCB(TPPTmisc, "sosend so resid iovcnt", so,
                    729:                                uio->uio_resid, uio->uio_iovcnt, 0);
                    730:                }
                    731:        ENDTRACE
                    732: }
                    733: 
                    734: tp_confirm(tpcb)
                    735: register struct tp_pcb *tpcb;
                    736: {
                    737:        struct tp_event E;
                    738:        if (tpcb->tp_state == TP_CONFIRMING)
                    739:            return DoEvent(T_ACPT_req);
                    740:        printf("Tp confirm called when not confirming; tpcb 0x%x, state 0x%x\n",
                    741:                tpcb, tpcb->tp_state);
                    742:        return 0;
                    743: }
                    744: 
                    745: /*
                    746:  * Process control data sent with sendmsg()
                    747:  */
                    748: tp_snd_control(m, so, data)
                    749:        struct mbuf *m;
                    750:        struct socket *so;
                    751:        register struct mbuf **data;
                    752: {
                    753:        register struct cmsghdr *ch;
                    754:        int error = 0;
                    755: 
                    756:        if (m && m->m_len) {
                    757:                ch = mtod(m, struct cmsghdr *);
                    758:                m->m_len -= sizeof (*ch);
                    759:                m->m_data += sizeof (*ch);
                    760:                error = tp_ctloutput(PRCO_SETOPT,
                    761:                                                         so, ch->cmsg_level, ch->cmsg_type, &m);
                    762:                if (ch->cmsg_type == TPOPT_DISC_DATA) {
                    763:                        if (data && *data) {
                    764:                                m_freem(*data);
                    765:                                *data = 0;
                    766:                        }
                    767:                        error = tp_usrreq(so, PRU_DISCONNECT, (struct mbuf *)0,
                    768:                                                                (caddr_t)0, (struct mbuf *)0);
                    769:                }
                    770:        }
                    771:        if (m)
                    772:                m_freem(m);
                    773:        return error;
                    774: }

unix.superglobalmegacorp.com

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