Annotation of XNU/bsd/netiso/tp_subr2.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_subr2.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:  * Some auxiliary routines:
                     87:  *     tp_protocol_error: required by xebec- called when a combo of state,
                     88:  *         event, predicate isn't covered for by the transition file.
                     89:  *     tp_indicate: gives indications(signals) to the user process
                     90:  *     tp_getoptions: initializes variables that are affected by the options
                     91:  *         chosen.
                     92:  */
                     93: 
                     94: /* this def'n is to cause the expansion of this macro in the
                     95:  * routine tp_local_credit :
                     96:  */
                     97: #define LOCAL_CREDIT_EXPAND
                     98: 
                     99: #include <sys/param.h>
                    100: #include <sys/systm.h>
                    101: #include <sys/mbuf.h>
                    102: #include <sys/socket.h>
                    103: #include <sys/socketvar.h>
                    104: #include <sys/domain.h>
                    105: #include <sys/protosw.h>
                    106: #include <sys/errno.h>
                    107: #include <sys/time.h>
                    108: #include <sys/kernel.h>
                    109: 
                    110: #undef MNULL
                    111: #include <netiso/argo_debug.h>
                    112: #include <netiso/tp_param.h>
                    113: #include <netiso/tp_ip.h>
                    114: #include <netiso/iso.h>
                    115: #include <netiso/iso_errno.h>
                    116: #include <netiso/iso_pcb.h>
                    117: #include <netiso/tp_timer.h>
                    118: #include <netiso/tp_stat.h>
                    119: #include <netiso/tp_tpdu.h>
                    120: #include <netiso/tp_pcb.h>
                    121: #include <netiso/tp_seq.h>
                    122: #include <netiso/tp_trace.h>
                    123: #include <netiso/tp_user.h>
                    124: #include <netiso/cons.h>
                    125: 
                    126: #include <net/if.h>
                    127: #include <net/if_types.h>
                    128: #ifdef TRUE
                    129: #undef FALSE
                    130: #undef TRUE
                    131: #endif
                    132: #include <netccitt/x25.h>
                    133: #include <netccitt/pk.h>
                    134: #include <netccitt/pk_var.h>
                    135: 
                    136: void tp_rsyset();
                    137: 
                    138: /*
                    139:  * NAME:       tp_local_credit()
                    140:  *
                    141:  * CALLED FROM:
                    142:  *  tp_emit(), tp_usrreq()
                    143:  *
                    144:  * FUNCTION and ARGUMENTS:
                    145:  *     Computes the local credit and stashes it in tpcb->tp_lcredit.
                    146:  *  It's a macro in the production system rather than a procdure.
                    147:  *
                    148:  * RETURNS:
                    149:  *
                    150:  * SIDE EFFECTS:
                    151:  *
                    152:  * NOTES:
                    153:  *  This doesn't actually get called in a production system - 
                    154:  *  the macro gets expanded instead in place of calls to this proc.
                    155:  *  But for debugging, we call this and that allows us to add
                    156:  *  debugging messages easily here.
                    157:  */
                    158: void
                    159: tp_local_credit(tpcb)
                    160:        struct tp_pcb *tpcb;
                    161: {
                    162:        LOCAL_CREDIT(tpcb);
                    163:        IFDEBUG(D_CREDIT)
                    164:                printf("ref 0x%x lcdt 0x%x l_tpdusize 0x%x decbit 0x%x\n",
                    165:                        tpcb->tp_lref, 
                    166:                        tpcb->tp_lcredit, 
                    167:                        tpcb->tp_l_tpdusize, 
                    168:                        tpcb->tp_decbit, 
                    169:                        tpcb->tp_cong_win
                    170:                        );
                    171:        ENDDEBUG
                    172:        IFTRACE(D_CREDIT)
                    173:                tptraceTPCB(TPPTmisc,
                    174:                        "lcdt tpdusz \n",
                    175:                         tpcb->tp_lcredit, tpcb->tp_l_tpdusize, 0, 0);
                    176:        ENDTRACE
                    177: }
                    178: 
                    179: /*
                    180:  * NAME:  tp_protocol_error()
                    181:  *
                    182:  * CALLED FROM:
                    183:  *  tp_driver(), when it doesn't know what to do with
                    184:  *     a combo of event, state, predicate
                    185:  *
                    186:  * FUNCTION and ARGUMENTS:
                    187:  *  print error mesg 
                    188:  *
                    189:  * RETURN VALUE:
                    190:  *  EIO - always
                    191:  *
                    192:  * SIDE EFFECTS:
                    193:  *
                    194:  * NOTES:
                    195:  */
                    196: int
                    197: tp_protocol_error(e,tpcb)
                    198:        struct tp_event *e;
                    199:        struct tp_pcb   *tpcb;
                    200: {
                    201:        printf("TP PROTOCOL ERROR! tpcb 0x%x event 0x%x, state 0x%x\n",
                    202:                tpcb, e->ev_number, tpcb->tp_state);
                    203:        IFTRACE(D_DRIVER)
                    204:                tptraceTPCB(TPPTmisc, "PROTOCOL ERROR tpcb event state",
                    205:                        tpcb, e->ev_number, tpcb->tp_state, 0 );
                    206:        ENDTRACE
                    207:        return EIO; /* for lack of anything better */
                    208: }
                    209: 
                    210: 
                    211: /* Not used at the moment */
                    212: ProtoHook
                    213: tp_drain()
                    214: {
                    215:        return 0;
                    216: }
                    217: 
                    218: 
                    219: /*
                    220:  * NAME: tp_indicate()
                    221:  *
                    222:  * CALLED FROM:
                    223:  *     tp.trans when XPD arrive, when a connection is being disconnected by
                    224:  *  the arrival of a DR or ER, and when a connection times out.
                    225:  *
                    226:  * FUNCTION and ARGUMENTS:
                    227:  *  (ind) is the type of indication : T_DISCONNECT, T_XPD
                    228:  *  (error) is an E* value that will be put in the socket structure
                    229:  *  to be passed along to the user later.
                    230:  *     Gives a SIGURG to the user process or group indicated by the socket
                    231:  *     attached to the tpcb.
                    232:  *
                    233:  * RETURNS:  Rien
                    234:  * 
                    235:  * SIDE EFFECTS:
                    236:  *
                    237:  * NOTES:
                    238:  */
                    239: void
                    240: tp_indicate(ind, tpcb, error)
                    241:        int                             ind; 
                    242:        u_short                 error;
                    243:        register struct tp_pcb  *tpcb;
                    244: {
                    245:        register struct socket *so = tpcb->tp_sock;
                    246:        IFTRACE(D_INDICATION)
                    247:                tptraceTPCB(TPPTindicate, ind, *(u_short *)(tpcb->tp_lsuffix), 
                    248:                        *(u_short *)(tpcb->tp_fsuffix), error,so->so_pgid);
                    249:        ENDTRACE
                    250:        IFDEBUG(D_INDICATION)
                    251:                char *ls, *fs;
                    252:                ls = tpcb->tp_lsuffix, 
                    253:                fs = tpcb->tp_fsuffix, 
                    254: 
                    255:                printf(
                    256: "indicate 0x%x lsuf 0x%02x%02x fsuf 0x%02x%02x err 0x%x  noind 0x%x ref 0x%x\n",
                    257:                ind, 
                    258:                *ls, *(ls+1), *fs, *(fs+1),
                    259:                error, /*so->so_pgrp,*/
                    260:                tpcb->tp_no_disc_indications,
                    261:                tpcb->tp_lref);
                    262:        ENDDEBUG
                    263: 
                    264:        if (ind == ER_TPDU) {
                    265:                register struct mbuf *m;
                    266:                struct tp_disc_reason x;
                    267: 
                    268:                if ((so->so_state & SS_CANTRCVMORE) == 0 &&
                    269:                                (m = m_get(M_DONTWAIT, MT_OOBDATA)) != 0) {
                    270: 
                    271:                        x.dr_hdr.cmsg_len = m->m_len = sizeof(x);
                    272:                        x.dr_hdr.cmsg_level = SOL_TRANSPORT;
                    273:                        x.dr_hdr.cmsg_type= TPOPT_DISC_REASON;
                    274:                        x.dr_reason = error;
                    275:                        *mtod(m, struct tp_disc_reason *) = x;
                    276:                        sbappendrecord(&tpcb->tp_Xrcv, m);
                    277:                        error = 0;
                    278:                } else
                    279:                        error = ECONNRESET;
                    280:        }
                    281:        so->so_error = error;
                    282: 
                    283:        if (ind == T_DISCONNECT)  {
                    284:                if (error == 0)
                    285:                        so->so_error = ENOTCONN;
                    286:                if ( tpcb->tp_no_disc_indications )
                    287:                        return;
                    288:        }
                    289:        IFTRACE(D_INDICATION)
                    290:                tptraceTPCB(TPPTmisc, "doing sohasoutofband(so)", so,0,0,0);
                    291:        ENDTRACE
                    292:        sohasoutofband(so);
                    293: }
                    294: 
                    295: /*
                    296:  * NAME : tp_getoptions()
                    297:  *
                    298:  * CALLED FROM:
                    299:  *     tp.trans whenever we go into OPEN state 
                    300:  *
                    301:  * FUNCTION and ARGUMENTS:
                    302:  *  sets the proper flags and values in the tpcb, to control
                    303:  *  the appropriate actions for the given class, options,
                    304:  *  sequence space, etc, etc.
                    305:  * 
                    306:  * RETURNS: Nada
                    307:  * 
                    308:  * SIDE EFFECTS:
                    309:  *
                    310:  * NOTES:
                    311:  */
                    312: void
                    313: tp_getoptions(tpcb)
                    314: struct tp_pcb *tpcb;
                    315: {
                    316:        tpcb->tp_seqmask = 
                    317:                tpcb->tp_xtd_format ?   TP_XTD_FMT_MASK :       TP_NML_FMT_MASK ;
                    318:        tpcb->tp_seqbit =
                    319:                tpcb->tp_xtd_format ?   TP_XTD_FMT_BIT :        TP_NML_FMT_BIT ;
                    320:        tpcb->tp_seqhalf = tpcb->tp_seqbit >> 1;
                    321:        tpcb->tp_dt_ticks =
                    322:                max(tpcb->tp_dt_ticks, (tpcb->tp_peer_acktime + 2));
                    323:        tp_rsyset(tpcb);
                    324:        
                    325: }
                    326: 
                    327: /*
                    328:  * NAME:  tp_recycle_tsuffix()
                    329:  *
                    330:  * CALLED FROM:
                    331:  *  Called when a ref is frozen.
                    332:  *
                    333:  * FUNCTION and ARGUMENTS:
                    334:  *  allows the suffix to be reused. 
                    335:  *
                    336:  * RETURNS: zilch
                    337:  *
                    338:  * SIDE EFFECTS:
                    339:  *
                    340:  * NOTES:
                    341:  */
                    342: void
                    343: tp_recycle_tsuffix(tpcb)
                    344:        struct tp_pcb   *tpcb;
                    345: {
                    346:        bzero((caddr_t)tpcb->tp_lsuffix, sizeof( tpcb->tp_lsuffix));
                    347:        bzero((caddr_t)tpcb->tp_fsuffix, sizeof( tpcb->tp_fsuffix));
                    348:        tpcb->tp_fsuffixlen = tpcb->tp_lsuffixlen = 0;
                    349: 
                    350:        (tpcb->tp_nlproto->nlp_recycle_suffix)(tpcb->tp_npcb);
                    351: }
                    352: 
                    353: /*
                    354:  * NAME: tp_quench()
                    355:  *
                    356:  * CALLED FROM:
                    357:  *  tp{af}_quench() when ICMP source quench or similar thing arrives.
                    358:  *
                    359:  * FUNCTION and ARGUMENTS:
                    360:  *  Drop the congestion window back to 1.
                    361:  *  Congestion window scheme:
                    362:  *  Initial value is 1.  ("slow start" as Nagle, et. al. call it)
                    363:  *  For each good ack that arrives, the congestion window is increased
                    364:  *  by 1 (up to max size of logical infinity, which is to say, 
                    365:  *     it doesn't wrap around).
                    366:  *  Source quench causes it to drop back to 1.
                    367:  *  tp_send() uses the smaller of (regular window, congestion window). 
                    368:  *  One retransmission strategy option is to have any retransmission 
                    369:  *     cause reset the congestion window back  to 1.
                    370:  *
                    371:  *     (cmd) is either PRC_QUENCH: source quench, or
                    372:  *             PRC_QUENCH2: dest. quench (dec bit)
                    373:  *
                    374:  * RETURNS:
                    375:  * 
                    376:  * SIDE EFFECTS:
                    377:  * 
                    378:  * NOTES:
                    379:  */
                    380: void
                    381: tp_quench( tpcb, cmd )
                    382:        struct tp_pcb *tpcb;
                    383:        int cmd;
                    384: {
                    385:        IFDEBUG(D_QUENCH)
                    386:                printf("tp_quench tpcb 0x%x ref 0x%x sufx 0x%x\n",
                    387:                        tpcb, tpcb->tp_lref, *(u_short *)(tpcb->tp_lsuffix));
                    388:                printf("cong_win 0x%x decbit 0x%x \n",
                    389:                        tpcb->tp_cong_win, tpcb->tp_decbit);
                    390:        ENDDEBUG
                    391:        switch(cmd) {
                    392:                case PRC_QUENCH:
                    393:                        tpcb->tp_cong_win = tpcb->tp_l_tpdusize;
                    394:                        IncStat(ts_quench);
                    395:                        break;
                    396:                case PRC_QUENCH2:
                    397:                        tpcb->tp_cong_win = tpcb->tp_l_tpdusize; /* might as well quench source also */
                    398:                        tpcb->tp_decbit = TP_DECBIT_CLEAR_COUNT;
                    399:                        IncStat(ts_rcvdecbit);
                    400:                        break;
                    401:        }
                    402: }
                    403: 
                    404: 
                    405: /*
                    406:  * NAME:       tp_netcmd()
                    407:  *
                    408:  * CALLED FROM:                        
                    409:  *
                    410:  * FUNCTION and ARGUMENTS:                     
                    411:  *
                    412:  * RETURNS:                    
                    413:  *
                    414:  * SIDE EFFECTS:       
                    415:  *
                    416:  * NOTES:                      
                    417:  */
                    418: tp_netcmd( tpcb, cmd )
                    419:        struct tp_pcb *tpcb;
                    420:        int cmd;
                    421: {
                    422: #if TPCONS
                    423:        struct isopcb *isop;
                    424:        struct pklcd *lcp;
                    425: 
                    426:        if (tpcb->tp_netservice != ISO_CONS)
                    427:                return;
                    428:        isop = (struct isopcb *)tpcb->tp_npcb;
                    429:        lcp = (struct pklcd *)isop->isop_chan;
                    430:        switch (cmd) {
                    431: 
                    432:        case CONN_CLOSE:
                    433:        case CONN_REFUSE:
                    434:                if (isop->isop_refcnt == 1) {
                    435:                        /* This is really superfluous, since it would happen
                    436:                           anyway in iso_pcbdetach, although it is a courtesy
                    437:                           to free up the x.25 channel before the refwait timer
                    438:                           expires. */
                    439:                        lcp->lcd_upper = 0;
                    440:                        lcp->lcd_upnext = 0;
                    441:                        pk_disconnect(lcp);
                    442:                        isop->isop_chan = 0;
                    443:                        isop->isop_refcnt = 0;
                    444:                }
                    445:                break;
                    446: 
                    447:        default:
                    448:                printf("tp_netcmd(0x%x, 0x%x) NOT IMPLEMENTED\n", tpcb, cmd);
                    449:                break;
                    450:        }
                    451: #else /* TPCONS */
                    452:        printf("tp_netcmd(): X25 NOT CONFIGURED!!\n");
                    453: #endif
                    454: }
                    455: /*
                    456:  * CALLED FROM:
                    457:  *  tp_ctloutput() and tp_emit()
                    458:  * FUNCTION and ARGUMENTS:
                    459:  *     Convert a class mask to the highest numeric value it represents.
                    460:  */
                    461: 
                    462: int
                    463: tp_mask_to_num(x)
                    464:        u_char x;
                    465: {
                    466:        register int j;
                    467: 
                    468:        for(j = 4; j>=0 ;j--) {
                    469:                if(x & (1<<j))
                    470:                        break;
                    471:        }
                    472:        ASSERT( (j == 4) || (j == 0) ); /* for now */
                    473:        if( (j != 4) && (j != 0) ) {
                    474:                printf("ASSERTION ERROR: tp_mask_to_num: x 0x%x j %d\n",
                    475:                        x, j);
                    476:        }
                    477:        IFTRACE(D_TPINPUT)
                    478:                tptrace(TPPTmisc, "tp_mask_to_num(x) returns j", x, j, 0, 0);
                    479:        ENDTRACE
                    480:        IFDEBUG(D_TPINPUT)
                    481:                printf("tp_mask_to_num(0x%x) returns 0x%x\n", x, j);
                    482:        ENDDEBUG
                    483:        return j;
                    484: }
                    485: 
                    486: static 
                    487: copyQOSparms(src, dst)
                    488:        struct tp_conn_param *src, *dst;
                    489: {
                    490:        /* copy all but the bits stuff at the end */
                    491: #define COPYSIZE (12 * sizeof(short))
                    492: 
                    493:        bcopy((caddr_t)src, (caddr_t)dst, COPYSIZE);
                    494:        dst->p_tpdusize = src->p_tpdusize;
                    495:        dst->p_ack_strat = src->p_ack_strat;
                    496:        dst->p_rx_strat = src->p_rx_strat;
                    497: #undef COPYSIZE
                    498: }
                    499: /*
                    500:  * Determine a reasonable value for maxseg size.
                    501:  * If the route is known, check route for mtu.
                    502:  * We also initialize the congestion/slow start
                    503:  * window to be a single segment if the destination isn't local.
                    504:  * While looking at the routing entry, we also initialize other path-dependent
                    505:  * parameters from pre-set or cached values in the routing entry.
                    506:  */
                    507: void
                    508: tp_mss(tpcb, nhdr_size)
                    509:        register struct tp_pcb *tpcb;
                    510:        int nhdr_size;
                    511: {
                    512:        register struct rtentry *rt;
                    513:        struct ifnet *ifp;
                    514:        register int rtt, mss;
                    515:        u_long bufsize;
                    516:        int i, ssthresh = 0, rt_mss;
                    517:        struct socket *so;
                    518: 
                    519:        if (tpcb->tp_ptpdusize)
                    520:                mss = tpcb->tp_ptpdusize << 7;
                    521:        else
                    522:                mss = 1 << tpcb->tp_tpdusize;
                    523:        so = tpcb->tp_sock;
                    524:        if ((rt = *(tpcb->tp_routep)) == 0) {
                    525:                bufsize = so->so_rcv.sb_hiwat;
                    526:                goto punt_route;
                    527:        }
                    528:        ifp = rt->rt_ifp;
                    529: 
                    530: #ifdef RTV_MTU /* if route characteristics exist ... */
                    531:        /*
                    532:         * While we're here, check if there's an initial rtt
                    533:         * or rttvar.  Convert from the route-table units
                    534:         * to hz ticks for the smoothed timers and slow-timeout units
                    535:         * for other inital variables.
                    536:         */
                    537:        if (tpcb->tp_rtt == 0 && (rtt = rt->rt_rmx.rmx_rtt)) {
                    538:                tpcb->tp_rtt = rtt * hz / RTM_RTTUNIT;
                    539:                if (rt->rt_rmx.rmx_rttvar)
                    540:                        tpcb->tp_rtv = rt->rt_rmx.rmx_rttvar
                    541:                                                * hz / RTM_RTTUNIT;
                    542:                else
                    543:                        tpcb->tp_rtv = tpcb->tp_rtt;
                    544:        }
                    545:        /*
                    546:         * if there's an mtu associated with the route, use it
                    547:         */
                    548:        if (rt->rt_rmx.rmx_mtu)
                    549:                rt_mss = rt->rt_rmx.rmx_mtu - nhdr_size;
                    550:        else
                    551: #endif /* RTV_MTU */
                    552:                rt_mss = (ifp->if_mtu - nhdr_size);
                    553:        if (tpcb->tp_ptpdusize == 0 || /* assume application doesn't care */
                    554:            mss > rt_mss /* network won't support what was asked for */)
                    555:                mss = rt_mss;
                    556:        /* can propose mtu which are multiples of 128 */
                    557:        mss &= ~0x7f;
                    558:        /*
                    559:         * If there's a pipesize, change the socket buffer
                    560:         * to that size.
                    561:         */
                    562: #ifdef RTV_SPIPE
                    563:        if ((bufsize = rt->rt_rmx.rmx_sendpipe) > 0) {
                    564: #endif
                    565:                bufsize = min(bufsize, so->so_snd.sb_hiwat);
                    566:                (void) sbreserve(&so->so_snd, bufsize);
                    567:        }
                    568: #ifdef RTV_SPIPE
                    569:        if ((bufsize = rt->rt_rmx.rmx_recvpipe) > 0) {
                    570: #endif
                    571:                bufsize = min(bufsize, so->so_rcv.sb_hiwat);
                    572:                (void) sbreserve(&so->so_rcv, bufsize);
                    573:        } else
                    574:                bufsize = so->so_rcv.sb_hiwat;
                    575: #ifdef RTV_SSTHRESH
                    576:        /*
                    577:         * There's some sort of gateway or interface
                    578:         * buffer limit on the path.  Use this to set
                    579:         * the slow start threshhold, but set the
                    580:         * threshold to no less than 2*mss.
                    581:         */
                    582:        ssthresh = rt->rt_rmx.rmx_ssthresh;
                    583: punt_route:
                    584:        /*
                    585:         * The current mss is initialized to the default value.
                    586:         * If we compute a smaller value, reduce the current mss.
                    587:         * If we compute a larger value, return it for use in sending
                    588:         * a max seg size option.
                    589:         * If we received an offer, don't exceed it.
                    590:         * However, do not accept offers under 128 bytes.
                    591:         */
                    592:        if (tpcb->tp_l_tpdusize)
                    593:                mss = min(mss, tpcb->tp_l_tpdusize);
                    594:        /*
                    595:         * We want a minimum recv window of 4 packets to
                    596:         * signal packet loss by duplicate acks.
                    597:         */
                    598:        mss = min(mss, bufsize >> 2) & ~0x7f;
                    599:        mss = max(mss, 128);            /* sanity */
                    600:        tpcb->tp_cong_win =
                    601:                (rt == 0 || (rt->rt_flags & RTF_GATEWAY)) ? mss : bufsize;
                    602:        tpcb->tp_l_tpdusize = mss;
                    603:        tp_rsyset(tpcb);
                    604:        tpcb->tp_ssthresh = max(2 * mss, ssthresh);
                    605:        /* Calculate log2 of mss */
                    606:        for (i = TP_MIN_TPDUSIZE + 1; i <= TP_MAX_TPDUSIZE; i++)
                    607:                if ((1 << i) > mss)
                    608:                        break;
                    609:        i--;
                    610:        tpcb->tp_tpdusize = i;
                    611: #endif /* RTV_MTU */
                    612: }
                    613: 
                    614: /*
                    615:  * CALLED FROM:
                    616:  *  tp_usrreq on PRU_CONNECT and tp_input on receipt of CR
                    617:  *     
                    618:  * FUNCTION and ARGUMENTS:
                    619:  *     -- An mbuf containing the peer's network address.
                    620:  *  -- Our control block, which will be modified
                    621:  *  -- In the case of cons, a control block for that layer.
                    622:  *
                    623:  *     
                    624:  * RETURNS:
                    625:  *     errno value      : 
                    626:  *     EAFNOSUPPORT if can't find an nl_protosw for x.25 (really could panic)
                    627:  *     ECONNREFUSED if trying to run TP0 with non-type 37 address
                    628:  *  possibly other E* returned from cons_netcmd()
                    629:  *
                    630:  * SIDE EFFECTS:
                    631:  *   Determines recommended tpdusize, buffering and intial delays
                    632:  *      based on information cached on the route.
                    633:  */
                    634: int
                    635: tp_route_to( m, tpcb, channel)
                    636:        struct mbuf                                     *m;
                    637:        register struct tp_pcb          *tpcb;
                    638:        caddr_t                                         channel;
                    639: {
                    640:        register struct sockaddr_iso *siso;     /* NOTE: this may be a sockaddr_in */
                    641:        extern struct tp_conn_param tp_conn_param[];
                    642:        int error = 0, save_netservice = tpcb->tp_netservice;
                    643:        register struct rtentry *rt = 0;
                    644:        int nhdr_size, mtu, bufsize;
                    645: 
                    646:        siso = mtod(m, struct sockaddr_iso *);
                    647:        IFTRACE(D_CONN)
                    648:                tptraceTPCB(TPPTmisc, 
                    649:                "route_to: so  afi netservice class",
                    650:                tpcb->tp_sock, siso->siso_addr.isoa_genaddr[0], tpcb->tp_netservice,
                    651:                        tpcb->tp_class);
                    652:        ENDTRACE
                    653:        IFDEBUG(D_CONN)
                    654:                printf("tp_route_to( m x%x, channel 0x%x, tpcb 0x%x netserv 0x%x)\n", 
                    655:                        m, channel, tpcb, tpcb->tp_netservice);
                    656:                printf("m->mlen x%x, m->m_data:\n", m->m_len);
                    657:                dump_buf(mtod(m, caddr_t), m->m_len);
                    658:        ENDDEBUG
                    659:        if (channel) {
                    660: #if TPCONS
                    661:                struct pklcd *lcp = (struct pklcd *)channel;
                    662:                struct isopcb *isop = (struct isopcb *)lcp->lcd_upnext,
                    663:                        *isop_new = (struct isopcb *)tpcb->tp_npcb;
                    664:                /* The next 2 lines believe that you haven't
                    665:                   set any network level options or done a pcbconnect
                    666:                   and XXXXXXX'edly apply to both inpcb's and isopcb's */
                    667:                remque(isop_new);
                    668:                FREE(isop_new, M_PCB);
                    669:                tpcb->tp_npcb = (caddr_t)isop;
                    670:                tpcb->tp_netservice = ISO_CONS;
                    671:                tpcb->tp_nlproto = nl_protosw + ISO_CONS;
                    672:                if (isop->isop_refcnt++ == 0) {
                    673:                        iso_putsufx(isop, tpcb->tp_lsuffix, tpcb->tp_lsuffixlen, TP_LOCAL);
                    674:                        isop->isop_socket = tpcb->tp_sock;
                    675:                } else
                    676:                        /* there are already connections sharing this */;
                    677: #endif
                    678:        } else {
                    679:                switch (siso->siso_family) {
                    680:                default:
                    681:                        error = EAFNOSUPPORT;
                    682:                        goto done;
                    683: #if ISO
                    684:                case AF_ISO:
                    685:                {
                    686:                        struct isopcb *isop = (struct isopcb *)tpcb->tp_npcb;
                    687:                        int flags = tpcb->tp_sock->so_options & SO_DONTROUTE;
                    688:                        tpcb->tp_netservice = ISO_CLNS;
                    689:                        if (clnp_route(&siso->siso_addr, &isop->isop_route,
                    690:                                                        flags, (void **)0, (void **)0) == 0) {
                    691:                                rt = isop->isop_route.ro_rt;
                    692:                                if (rt && rt->rt_flags & RTF_PROTO1)
                    693:                                        tpcb->tp_netservice = ISO_CONS;
                    694:                        }
                    695:                }    break;
                    696: #endif
                    697: #if INET
                    698:                case AF_INET:
                    699:                        tpcb->tp_netservice = IN_CLNS;
                    700: #endif
                    701:                }
                    702:                if (tpcb->tp_nlproto->nlp_afamily != siso->siso_family) {
                    703:                        IFDEBUG(D_CONN)
                    704:                                printf("tp_route_to( CHANGING nlproto old 0x%x new 0x%x)\n", 
                    705:                                                save_netservice, tpcb->tp_netservice);
                    706:                        ENDDEBUG
                    707:                        if (error = tp_set_npcb(tpcb))
                    708:                                goto done;
                    709:                }
                    710:                IFDEBUG(D_CONN)
                    711:                        printf("tp_route_to  calling nlp_pcbconn, netserv %d\n",
                    712:                                tpcb->tp_netservice);
                    713:                ENDDEBUG
                    714:                tpcb->tp_nlproto = nl_protosw + tpcb->tp_netservice;
                    715:                error = (tpcb->tp_nlproto->nlp_pcbconn)(tpcb->tp_npcb, m);
                    716:        }
                    717:        if (error)
                    718:                goto done;
                    719:        nhdr_size = tpcb->tp_nlproto->nlp_mtu(tpcb); /* only gets common info */
                    720:        tp_mss(tpcb, nhdr_size);
                    721: done:
                    722:        IFDEBUG(D_CONN)
                    723:                printf("tp_route_to  returns 0x%x\n", error);
                    724:        ENDDEBUG
                    725:        IFTRACE(D_CONN)
                    726:                tptraceTPCB(TPPTmisc, "route_to: returns: error netserv class", error, 
                    727:                        tpcb->tp_netservice, tpcb->tp_class, 0);
                    728:        ENDTRACE
                    729:        return error;
                    730: }
                    731: 
                    732: 
                    733: /* class zero version */
                    734: void
                    735: tp0_stash( tpcb, e )
                    736:        register struct tp_pcb          *tpcb;
                    737:        register struct tp_event        *e;
                    738: {
                    739: #ifndef lint
                    740: #define E e->ATTR(DT_TPDU)
                    741: #else /* lint */
                    742: #define E e->ev_union.EV_DT_TPDU
                    743: #endif /* lint */
                    744: 
                    745:        register struct sockbuf *sb = &tpcb->tp_sock->so_rcv;
                    746:        register struct isopcb *isop = (struct isopcb *)tpcb->tp_npcb;
                    747: 
                    748:        IFPERF(tpcb)
                    749:                PStat(tpcb, Nb_from_ll) += E.e_datalen;
                    750:                tpmeas(tpcb->tp_lref, TPtime_from_ll, &e->e_time,
                    751:                                E.e_seq, PStat(tpcb, Nb_from_ll), E.e_datalen);
                    752:        ENDPERF
                    753: 
                    754:        IFDEBUG(D_STASH)
                    755:                printf("stash EQ: seq 0x%x datalen 0x%x eot 0x%x", 
                    756:                E.e_seq, E.e_datalen, E.e_eot);
                    757:        ENDDEBUG
                    758: 
                    759:        IFTRACE(D_STASH)
                    760:                tptraceTPCB(TPPTmisc, "stash EQ: seq len eot", 
                    761:                E.e_seq, E.e_datalen, E.e_eot, 0);
                    762:        ENDTRACE
                    763: 
                    764:        if ( E.e_eot ) {
                    765:                register struct mbuf *n = E.e_data;
                    766:                n->m_flags |= M_EOR;
                    767:                n->m_act = MNULL; /* set on tp_input */
                    768:        }
                    769:        sbappend(sb, E.e_data);
                    770:        IFDEBUG(D_STASH)
                    771:                dump_mbuf(sb->sb_mb, "stash 0: so_rcv after appending");
                    772:        ENDDEBUG
                    773:        if (tpcb->tp_netservice != ISO_CONS)
                    774:                printf("tp0_stash: tp running over something wierd\n");
                    775:        else {
                    776:                register struct pklcd *lcp = (struct pklcd *)isop->isop_chan;
                    777:                pk_flowcontrol(lcp, sbspace(sb) <= 0, 1);
                    778:        }
                    779: } 
                    780: 
                    781: void
                    782: tp0_openflow(tpcb)
                    783: register struct tp_pcb *tpcb;
                    784: {
                    785:        register struct isopcb *isop = (struct isopcb *)tpcb->tp_npcb;
                    786:        if (tpcb->tp_netservice != ISO_CONS)
                    787:                printf("tp0_openflow: tp running over something wierd\n");
                    788:        else {
                    789:                register struct pklcd *lcp = (struct pklcd *)isop->isop_chan;
                    790:                if (lcp->lcd_rxrnr_condition)
                    791:                        pk_flowcontrol(lcp, 0, 0);
                    792:        }
                    793: }
                    794: #ifndef TPCONS
                    795: static
                    796: pk_flowcontrol() {}
                    797: #endif
                    798: 
                    799: #ifdef TP_PERF_MEAS
                    800: /*
                    801:  * CALLED FROM:
                    802:  *  tp_ctloutput() when the user sets TPOPT_PERF_MEAS on
                    803:  *  and tp_newsocket() when a new connection is made from 
                    804:  *  a listening socket with tp_perf_on == true.
                    805:  * FUNCTION and ARGUMENTS:
                    806:  *  (tpcb) is the usual; this procedure gets a clear cluster mbuf for
                    807:  *  a tp_pmeas structure, and makes tpcb->tp_p_meas point to it.
                    808:  * RETURN VALUE:
                    809:  *  ENOBUFS if it cannot get a cluster mbuf.
                    810:  */
                    811: 
                    812: int 
                    813: tp_setup_perf(tpcb)
                    814:        register struct tp_pcb *tpcb;
                    815: {
                    816:        register struct mbuf *q;
                    817: 
                    818:        if( tpcb->tp_p_meas == 0 ) {
                    819:                MGET(q, M_WAIT, MT_PCB);
                    820:                if (q == 0)
                    821:                        return ENOBUFS;
                    822:                MCLGET(q, M_WAIT);
                    823:                if ((q->m_flags & M_EXT) == 0) {
                    824:                        (void) m_free(q);
                    825:                        return ENOBUFS;
                    826:                }
                    827:                q->m_len = sizeof (struct tp_pmeas);
                    828:                tpcb->tp_p_mbuf = q;
                    829:                tpcb->tp_p_meas = mtod(q, struct tp_pmeas *);
                    830:                bzero( (caddr_t)tpcb->tp_p_meas, sizeof (struct tp_pmeas) );
                    831:                IFDEBUG(D_PERF_MEAS)
                    832:                        printf(
                    833:                        "tpcb 0x%x so 0x%x ref 0x%x tp_p_meas 0x%x tp_perf_on 0x%x\n", 
                    834:                                tpcb, tpcb->tp_sock, tpcb->tp_lref, 
                    835:                                tpcb->tp_p_meas, tpcb->tp_perf_on);
                    836:                ENDDEBUG
                    837:                tpcb->tp_perf_on = 1;
                    838:        }
                    839:        return 0;
                    840: }
                    841: #endif /* TP_PERF_MEAS */
                    842: 
                    843: #ifdef ARGO_DEBUG
                    844: dump_addr (addr)
                    845:        register struct sockaddr *addr;
                    846: {
                    847:        switch( addr->sa_family ) {
                    848:                case AF_INET:
                    849:                        dump_inaddr((struct sockaddr_in *)addr);
                    850:                        break;
                    851: #if ISO
                    852:                case AF_ISO:
                    853:                        dump_isoaddr((struct sockaddr_iso *)addr);
                    854:                        break;
                    855: #endif /* ISO */
                    856:                default:
                    857:                        printf("BAD AF: 0x%x\n", addr->sa_family);
                    858:                        break;
                    859:        }
                    860: }
                    861: 
                    862: #define        MAX_COLUMNS     8
                    863: /*
                    864:  *     Dump the buffer to the screen in a readable format. Format is:
                    865:  *
                    866:  *             hex/dec  where hex is the hex format, dec is the decimal format.
                    867:  *             columns of hex/dec numbers will be printed, followed by the
                    868:  *             character representations (if printable).
                    869:  */
                    870: Dump_buf(buf, len)
                    871: caddr_t        buf;
                    872: int            len;
                    873: {
                    874:        int             i,j;
                    875: #define Buf ((u_char *)buf)
                    876:        printf("Dump buf 0x%x len 0x%x\n", buf, len);
                    877:        for (i = 0; i < len; i += MAX_COLUMNS) {
                    878:                printf("+%d:\t", i);
                    879:                for (j = 0; j < MAX_COLUMNS; j++) {
                    880:                        if (i + j < len) {
                    881:                                printf("%x/%d\t", Buf[i+j], Buf[i+j]);
                    882:                        } else {
                    883:                                printf("        ");
                    884:                        }
                    885:                }
                    886: 
                    887:                for (j = 0; j < MAX_COLUMNS; j++) {
                    888:                        if (i + j < len) {
                    889:                                if (((Buf[i+j]) > 31) && ((Buf[i+j]) < 128))
                    890:                                        printf("%c", Buf[i+j]);
                    891:                                else
                    892:                                        printf(".");
                    893:                        }
                    894:                }
                    895:                printf("\n");
                    896:        }
                    897: }
                    898: #endif /* ARGO_DEBUG */

unix.superglobalmegacorp.com

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