Annotation of XNU/bsd/netiso/tp_subr2.c, revision 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.