Annotation of 43BSDReno/sys/netiso/tp_subr2.c, revision 1.1.1.1

1.1       root        1: /***********************************************************
                      2:                Copyright IBM Corporation 1987
                      3: 
                      4:                       All Rights Reserved
                      5: 
                      6: Permission to use, copy, modify, and distribute this software and its 
                      7: documentation for any purpose and without fee is hereby granted, 
                      8: provided that the above copyright notice appear in all copies and that
                      9: both that copyright notice and this permission notice appear in 
                     10: supporting documentation, and that the name of IBM not be
                     11: used in advertising or publicity pertaining to distribution of the
                     12: software without specific, written prior permission.  
                     13: 
                     14: IBM DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
                     15: ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
                     16: IBM BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
                     17: ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
                     18: WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
                     19: ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
                     20: SOFTWARE.
                     21: 
                     22: ******************************************************************/
                     23: 
                     24: /*
                     25:  * ARGO Project, Computer Sciences Dept., University of Wisconsin - Madison
                     26:  */
                     27: /* 
                     28:  * ARGO TP
                     29:  *
                     30:  * $Header: tp_subr2.c,v 5.5 88/11/18 17:28:55 nhall Exp $
                     31:  * $Source: /usr/argo/sys/netiso/RCS/tp_subr2.c,v $
                     32:  *     @(#)tp_subr2.c  7.4 (Berkeley) 9/22/89
                     33:  *
                     34:  * Some auxiliary routines:
                     35:  *             tp_protocol_error: required by xebec- called when a combo of state,
                     36:  *                     event, predicate isn't covered for by the transition file.
                     37:  *             tp_indicate: gives indications(signals) to the user process
                     38:  *             tp_getoptions: initializes variables that are affected by the options
                     39:  *          chosen.
                     40:  */
                     41: 
                     42: #ifndef lint
                     43: static char *rcsid = "$Header: tp_subr2.c,v 5.5 88/11/18 17:28:55 nhall Exp $";
                     44: #endif lint
                     45: 
                     46: #include "argoxtwentyfive.h"
                     47: 
                     48: /* this def'n is to cause the expansion of this macro in the
                     49:  * routine tp_local_credit :
                     50:  */
                     51: #define LOCAL_CREDIT_EXPAND
                     52: 
                     53: #include "param.h"
                     54: #include "mbuf.h"
                     55: #include "socket.h"
                     56: #include "socketvar.h"
                     57: #include "domain.h"
                     58: #include "protosw.h"
                     59: #include "errno.h"
                     60: #include "types.h"
                     61: #include "time.h"
                     62: #include "kernel.h"
                     63: #undef MNULL
                     64: #include "argo_debug.h"
                     65: #include "tp_param.h"
                     66: #include "tp_ip.h"
                     67: #include "iso.h"
                     68: #include "iso_errno.h"
                     69: #include "iso_pcb.h"
                     70: #include "tp_timer.h"
                     71: #include "tp_stat.h"
                     72: #include "tp_tpdu.h"
                     73: #include "tp_pcb.h"
                     74: #include "tp_seq.h"
                     75: #include "tp_trace.h"
                     76: #include "tp_user.h"
                     77: #include "cons.h"
                     78: 
                     79: /*
                     80:  * NAME:       tp_local_credit()
                     81:  *
                     82:  * CALLED FROM:
                     83:  *  tp_emit(), tp_usrreq()
                     84:  *
                     85:  * FUNCTION and ARGUMENTS:
                     86:  *     Computes the local credit and stashes it in tpcb->tp_lcredit.
                     87:  *  It's a macro in the production system rather than a procdure.
                     88:  *
                     89:  * RETURNS:
                     90:  *
                     91:  * SIDE EFFECTS:
                     92:  *
                     93:  * NOTES:
                     94:  *  This doesn't actually get called in a production system - 
                     95:  *  the macro gets expanded instead in place of calls to this proc.
                     96:  *  But for debugging, we call this and that allows us to add
                     97:  *  debugging messages easily here.
                     98:  */
                     99: void
                    100: tp_local_credit(tpcb)
                    101:        struct tp_pcb *tpcb;
                    102: {
                    103:        LOCAL_CREDIT(tpcb);
                    104:        IFDEBUG(D_CREDIT)
                    105:                printf("ref 0x%x lcdt 0x%x l_tpdusize 0x%x decbit 0x%x\n",
                    106:                        tpcb->tp_refp - tp_ref, 
                    107:                        tpcb->tp_lcredit, 
                    108:                        tpcb->tp_l_tpdusize, 
                    109:                        tpcb->tp_decbit, 
                    110:                        tpcb->tp_cong_win
                    111:                        );
                    112:        ENDDEBUG
                    113:        IFTRACE(D_CREDIT)
                    114:                tptraceTPCB(TPPTmisc,
                    115:                        "lcdt tpdusz \n",
                    116:                         tpcb->tp_lcredit, tpcb->tp_l_tpdusize, 0, 0);
                    117:        ENDTRACE
                    118: }
                    119: 
                    120: /*
                    121:  * NAME:  tp_protocol_error()
                    122:  *
                    123:  * CALLED FROM:
                    124:  *  tp_driver(), when it doesn't know what to do with
                    125:  *     a combo of event, state, predicate
                    126:  *
                    127:  * FUNCTION and ARGUMENTS:
                    128:  *  print error mesg 
                    129:  *
                    130:  * RETURN VALUE:
                    131:  *  EIO - always
                    132:  *
                    133:  * SIDE EFFECTS:
                    134:  *
                    135:  * NOTES:
                    136:  */
                    137: int
                    138: tp_protocol_error(e,tpcb)
                    139:        struct tp_event *e;
                    140:        struct tp_pcb   *tpcb;
                    141: {
                    142:        printf("TP PROTOCOL ERROR! tpcb 0x%x event 0x%x, state 0x%x\n",
                    143:                tpcb, e->ev_number, tpcb->tp_state);
                    144:        IFTRACE(D_DRIVER)
                    145:                tptraceTPCB(TPPTmisc, "PROTOCOL ERROR tpcb event state",
                    146:                        tpcb, e->ev_number, tpcb->tp_state, 0 );
                    147:        ENDTRACE
                    148:        return EIO; /* for lack of anything better */
                    149: }
                    150: 
                    151: 
                    152: /* Not used at the moment */
                    153: ProtoHook
                    154: tp_drain()
                    155: {
                    156:        return 0;
                    157: }
                    158: 
                    159: 
                    160: /*
                    161:  * NAME: tp_indicate()
                    162:  *
                    163:  * CALLED FROM:
                    164:  *     tp.trans when XPD arrive, when a connection is being disconnected by
                    165:  *  the arrival of a DR or ER, and when a connection times out.
                    166:  *
                    167:  * FUNCTION and ARGUMENTS:
                    168:  *  (ind) is the type of indication : T_DISCONNECT, T_XPD
                    169:  *  (error) is an E* value that will be put in the socket structure
                    170:  *  to be passed along to the user later.
                    171:  *     Gives a SIGURG to the user process or group indicated by the socket
                    172:  *     attached to the tpcb.
                    173:  *
                    174:  * RETURNS:  Rien
                    175:  * 
                    176:  * SIDE EFFECTS:
                    177:  *
                    178:  * NOTES:
                    179:  */
                    180: void
                    181: tp_indicate(ind, tpcb, error)
                    182:        int                             ind; 
                    183:        u_short                 error;
                    184:        register struct tp_pcb  *tpcb;
                    185: {
                    186:        register struct socket *so = tpcb->tp_sock;
                    187:        IFTRACE(D_INDICATION)
                    188:                tptraceTPCB(TPPTindicate, ind, *(u_short *)(tpcb->tp_lsuffix), 
                    189:                        *(u_short *)(tpcb->tp_fsuffix), error,so->so_pgid);
                    190:        ENDTRACE
                    191:        IFDEBUG(D_INDICATION)
                    192:                char *ls, *fs;
                    193:                ls = tpcb->tp_lsuffix, 
                    194:                fs = tpcb->tp_fsuffix, 
                    195: 
                    196:                printf(
                    197: "indicate 0x%x lsuf 0x%02x%02x fsuf 0x%02x%02x err 0x%x  noind 0x%x ref 0x%x\n",
                    198:                ind, 
                    199:                *ls, *(ls+1), *fs, *(fs+1),
                    200:                error, /*so->so_pgrp,*/
                    201:                tpcb->tp_no_disc_indications,
                    202:                tpcb->tp_lref);
                    203:        ENDDEBUG
                    204: 
                    205:        so->so_error = error;
                    206: 
                    207:        if (ind == T_DISCONNECT)  {
                    208:                if ( tpcb->tp_no_disc_indications )
                    209:                        return;
                    210:        }
                    211:        IFTRACE(D_INDICATION)
                    212:                tptraceTPCB(TPPTmisc, "doing sohasoutofband(so)", so,0,0,0);
                    213:        ENDTRACE
                    214:        sohasoutofband(so);
                    215: }
                    216: 
                    217: /*
                    218:  * NAME : tp_getoptions()
                    219:  *
                    220:  * CALLED FROM:
                    221:  *     tp.trans whenever we go into OPEN state 
                    222:  *
                    223:  * FUNCTION and ARGUMENTS:
                    224:  *  sets the proper flags and values in the tpcb, to control
                    225:  *  the appropriate actions for the given class, options,
                    226:  *  sequence space, etc, etc.
                    227:  * 
                    228:  * RETURNS: Nada
                    229:  * 
                    230:  * SIDE EFFECTS:
                    231:  *
                    232:  * NOTES:
                    233:  */
                    234: void
                    235: tp_getoptions(tpcb)
                    236: struct tp_pcb *tpcb;
                    237: {
                    238:        tpcb->tp_seqmask = 
                    239:                tpcb->tp_xtd_format ?   TP_XTD_FMT_MASK :       TP_NML_FMT_MASK ;
                    240:        tpcb->tp_seqbit =
                    241:                tpcb->tp_xtd_format ?   TP_XTD_FMT_BIT :        TP_NML_FMT_BIT ;
                    242:        tpcb->tp_seqhalf = tpcb->tp_seqbit >> 1;
                    243:        tpcb->tp_dt_ticks =
                    244:                MAX(tpcb->tp_dt_ticks, (tpcb->tp_peer_acktime + 2));
                    245: 
                    246: }
                    247: 
                    248: /*
                    249:  * NAME:  tp_recycle_tsuffix()
                    250:  *
                    251:  * CALLED FROM:
                    252:  *  Called when a ref is frozen.
                    253:  *
                    254:  * FUNCTION and ARGUMENTS:
                    255:  *  allows the suffix to be reused. 
                    256:  *
                    257:  * RETURNS: zilch
                    258:  *
                    259:  * SIDE EFFECTS:
                    260:  *
                    261:  * NOTES:
                    262:  */
                    263: void
                    264: tp_recycle_tsuffix(tpcb)
                    265:        struct tp_pcb   *tpcb;
                    266: {
                    267:        bzero((caddr_t)tpcb->tp_lsuffix, sizeof( tpcb->tp_lsuffix));
                    268:        bzero((caddr_t)tpcb->tp_fsuffix, sizeof( tpcb->tp_fsuffix));
                    269:        tpcb->tp_fsuffixlen = tpcb->tp_lsuffixlen = 0;
                    270: 
                    271:        (tpcb->tp_nlproto->nlp_recycle_suffix)(tpcb->tp_npcb);
                    272: }
                    273: 
                    274: /*
                    275:  * NAME: tp_quench()
                    276:  *
                    277:  * CALLED FROM:
                    278:  *  tp{af}_quench() when ICMP source quench or similar thing arrives.
                    279:  *
                    280:  * FUNCTION and ARGUMENTS:
                    281:  *  Drop the congestion window back to 1.
                    282:  *  Congestion window scheme:
                    283:  *  Initial value is 1.  ("slow start" as Nagle, et. al. call it)
                    284:  *  For each good ack that arrives, the congestion window is increased
                    285:  *  by 1 (up to max size of logical infinity, which is to say, 
                    286:  *     it doesn't wrap around).
                    287:  *  Source quench causes it to drop back to 1.
                    288:  *  tp_send() uses the smaller of (regular window, congestion window). 
                    289:  *  One retransmission strategy option is to have any retransmission 
                    290:  *     cause reset the congestion window back  to 1.
                    291:  *
                    292:  *     (cmd) is either PRC_QUENCH: source quench, or
                    293:  *             PRC_QUENCH2: dest. quench (dec bit)
                    294:  *
                    295:  * RETURNS:
                    296:  * 
                    297:  * SIDE EFFECTS:
                    298:  * 
                    299:  * NOTES:
                    300:  */
                    301: void
                    302: tp_quench( tpcb, cmd )
                    303:        struct tp_pcb *tpcb;
                    304:        int cmd;
                    305: {
                    306:        IFDEBUG(D_QUENCH)
                    307:                printf("tp_quench tpcb 0x%x ref 0x%x sufx 0x%x\n",
                    308:                        tpcb, tpcb->tp_lref, *(u_short *)(tpcb->tp_lsuffix));
                    309:                printf("cong_win 0x%x decbit 0x%x \n",
                    310:                        tpcb->tp_cong_win, tpcb->tp_decbit);
                    311:        ENDDEBUG
                    312:        switch(cmd) {
                    313:                case PRC_QUENCH:
                    314:                        tpcb->tp_cong_win = 1;
                    315:                        IncStat(ts_quench);
                    316:                        break;
                    317:                case PRC_QUENCH2:
                    318:                        tpcb->tp_cong_win = 1; /* might as well quench source also */
                    319:                        tpcb->tp_decbit = TP_DECBIT_CLEAR_COUNT;
                    320:                        IncStat(ts_rcvdecbit);
                    321:                        break;
                    322:        }
                    323: }
                    324: 
                    325: 
                    326: /*
                    327:  * NAME:       tp_netcmd()
                    328:  *
                    329:  * CALLED FROM:                        
                    330:  *
                    331:  * FUNCTION and ARGUMENTS:                     
                    332:  *
                    333:  * RETURNS:                    
                    334:  *
                    335:  * SIDE EFFECTS:       
                    336:  *
                    337:  * NOTES:                      
                    338:  */
                    339: tp_netcmd( tpcb, cmd )
                    340:        struct tp_pcb *tpcb;
                    341:        int cmd;
                    342: {
                    343: #if NARGOXTWENTYFIVE > 0
                    344:        switch (cmd) {
                    345: 
                    346:        case CONN_CLOSE:
                    347:        case CONN_REFUSE:
                    348:                cons_netcmd( cmd, tpcb->tp_npcb, 0, tpcb->tp_class == TP_CLASS_4);
                    349:                /* TODO: can this last param be replaced by 
                    350:                *       tpcb->tp_netserv != ISO_CONS?)
                    351:                */
                    352:                break;
                    353: 
                    354:        default:
                    355:                printf("tp_netcmd(0x%x, 0x%x) NOT IMPLEMENTED\n", tpcb, cmd);
                    356:                break;
                    357:        }
                    358: #else NARGOXTWENTYFIVE
                    359:        printf("tp_netcmd(): X25 NOT CONFIGURED!!\n");
                    360: #endif NARGOXTWENTYFIVE > 0
                    361: }
                    362: /*
                    363:  * CALLED FROM:
                    364:  *  tp_ctloutput() and tp_emit()
                    365:  * FUNCTION and ARGUMENTS:
                    366:  *     Convert a class mask to the highest numeric value it represents.
                    367:  */
                    368: 
                    369: int
                    370: tp_mask_to_num(x)
                    371:        u_char x;
                    372: {
                    373:        register int j;
                    374: 
                    375:        for(j = 4; j>=0 ;j--) {
                    376:                if(x & (1<<j))
                    377:                        break;
                    378:        }
                    379:        ASSERT( (j == 4) || (j == 0) ); /* for now */
                    380:        if( (j != 4) && (j != 0) ) {
                    381:                printf("ASSERTION ERROR: tp_mask_to_num: x 0x%x j %d\n",
                    382:                        x, j);
                    383:        }
                    384:        IFTRACE(D_TPINPUT)
                    385:                tptrace(TPPTmisc, "tp_mask_to_num(x) returns j", x, j, 0, 0);
                    386:        ENDTRACE
                    387:        IFDEBUG(D_TPINPUT)
                    388:                printf("tp_mask_to_num(0x%x) returns 0x%x\n", x, j);
                    389:        ENDDEBUG
                    390:        return j;
                    391: }
                    392: 
                    393: static 
                    394: copyQOSparms(src, dst)
                    395:        struct tp_conn_param *src, *dst;
                    396: {
                    397:        /* copy all but the bits stuff at the end */
                    398: #define COPYSIZE (12 * sizeof(short))
                    399: 
                    400:        bcopy((caddr_t)src, (caddr_t)dst, COPYSIZE);
                    401:        dst->p_tpdusize = src->p_tpdusize;
                    402:        dst->p_ack_strat = src->p_ack_strat;
                    403:        dst->p_rx_strat = src->p_rx_strat;
                    404: #undef COPYSIZE
                    405: }
                    406: 
                    407: /*
                    408:  * CALLED FROM:
                    409:  *  tp_usrreq on PRU_CONNECT and tp_input on receipt of CR
                    410:  *     
                    411:  * FUNCTION and ARGUMENTS:
                    412:  *     route directly to x.25 if the address is type 37 - GROT.
                    413:  *  furthermore, let TP0 handle only type-37 addresses
                    414:  *
                    415:  *     Since this assumes that its address argument is in a mbuf, the
                    416:  *     parameter was changed to reflect this assumtion. This also
                    417:  *     implies that an mbuf must be allocated when this is
                    418:  *     called from tp_input
                    419:  *     
                    420:  * RETURNS:
                    421:  *     errno value      : 
                    422:  *     EAFNOSUPPORT if can't find an nl_protosw for x.25 (really could panic)
                    423:  *     ECONNREFUSED if trying to run TP0 with non-type 37 address
                    424:  *  possibly other E* returned from cons_netcmd()
                    425:  * NOTE:
                    426:  *  Would like to eliminate as much of this as possible -- 
                    427:  *  only one set of defaults (let the user set the parms according
                    428:  *  to parameters provided in the directory service).
                    429:  *  Left here for now 'cause we don't yet have a clean way to handle
                    430:  *  it on the passive end.
                    431:  */
                    432: int
                    433: tp_route_to( m, tpcb, channel)
                    434:        struct mbuf                                     *m;
                    435:        register struct tp_pcb          *tpcb;
                    436:        u_int                                           channel;
                    437: {
                    438:        register struct sockaddr_iso *siso;     /* NOTE: this may be a sockaddr_in */
                    439:        extern struct tp_conn_param tp_conn_param[];
                    440:        int error = 0;
                    441:        int     vc_to_kill = 0; /* kludge */
                    442: 
                    443:        siso = mtod(m, struct sockaddr_iso *);
                    444:        IFTRACE(D_CONN)
                    445:                tptraceTPCB(TPPTmisc, 
                    446:                "route_to: so  afi netservice class",
                    447:                tpcb->tp_sock, siso->siso_addr.isoa_genaddr[0], tpcb->tp_netservice,
                    448:                        tpcb->tp_class);
                    449:        ENDTRACE
                    450:        IFDEBUG(D_CONN)
                    451:                printf("tp_route_to( m x%x, channel 0x%x, tpcb 0x%x netserv 0x%x)\n", 
                    452:                        m, channel, tpcb, tpcb->tp_netservice);
                    453:                printf("m->mlen x%x, m->m_data:\n", m->m_len);
                    454:                dump_buf(mtod(m, caddr_t), m->m_len);
                    455:        ENDDEBUG
                    456:        if( siso->siso_family != tpcb->tp_domain ) {
                    457:                error = EAFNOSUPPORT;
                    458:                goto done;
                    459:        }
                    460:        IFDEBUG(D_CONN)
                    461:                printf("tp_route_to  calling nlp_pcbconn, netserv %d\n",
                    462:                        tpcb->tp_netservice);
                    463:        ENDDEBUG
                    464:        error = (tpcb->tp_nlproto->nlp_pcbconn)(tpcb->tp_sock->so_pcb, m);
                    465:        if( error )
                    466:                goto done;
                    467: 
                    468:        {
                    469:                register int save_netservice = tpcb->tp_netservice;
                    470: 
                    471:                switch(tpcb->tp_netservice) {
                    472:                case ISO_COSNS:
                    473:                case ISO_CLNS:
                    474:                        /* This is a kludge but seems necessary so the passive end
                    475:                         * can get long enough timers. sigh.
                    476:                        if( siso->siso_addr.osinet_idi[1] == (u_char)IDI_OSINET )
                    477:                         */
                    478: #define        IDI_OSINET      0x0004  /* bcd of "0004" */     
                    479:                        if( siso->siso_addr.isoa_genaddr[2] == (char)IDI_OSINET ) {
                    480:                                if( tpcb->tp_dont_change_params == 0) {
                    481:                                        copyQOSparms( &tp_conn_param[ISO_COSNS],
                    482:                                                        &tpcb->_tp_param);
                    483:                                }
                    484:                                tpcb->tp_flags |= TPF_NLQOS_PDN;
                    485:                        }
                    486:                        /* drop through to IN_CLNS*/
                    487:                case IN_CLNS:
                    488:                        if (iso_localifa(siso))
                    489:                                tpcb->tp_flags |= TPF_PEER_ON_SAMENET;
                    490:                        if( (tpcb->tp_class & TP_CLASS_4)==0 ) {
                    491:                                error = EPROTOTYPE;
                    492:                                break;
                    493:                        } 
                    494:                        tpcb->tp_class = TP_CLASS_4;  /* IGNORE dont_change_parms */
                    495:                        break;
                    496: 
                    497:                case ISO_CONS:
                    498: #if NARGOXTWENTYFIVE > 0
                    499:                        tpcb->tp_flags |= TPF_NLQOS_PDN;
                    500:                        if( tpcb->tp_dont_change_params == 0 ) {
                    501:                                copyQOSparms( &tp_conn_param[ISO_CONS],
                    502:                                                        &tpcb->_tp_param);
                    503:                        }
                    504:                        /*
                    505:                         * for use over x.25 really need a small receive window,
                    506:                         * need to start slowly, need small max negotiable tpdu size,
                    507:                         * and need to use the congestion window to the max
                    508:                         * IGNORES tp_dont_change_params for these!
                    509:                         */
                    510:                        if( tpcb->tp_sock->so_snd.sb_hiwat > 512 ) {
                    511:                                (void) soreserve(tpcb->tp_sock, 512, 512 );/* GAG */
                    512:                        }
                    513:                        tpcb->tp_rx_strat =  TPRX_USE_CW;
                    514: 
                    515:                        if( (tpcb->tp_nlproto != &nl_protosw[ISO_CONS]) ) {
                    516:                                /* if the listener was restricting us to clns,
                    517:                                 * ( we never get here if the listener isn't af_iso )
                    518:                                 * refuse the connection :
                    519:                                 * but we don't have a way to restrict thus - it's
                    520:                                 * utterly permissive.
                    521:                                        if(channel)  {
                    522:                                                (void) cons_netcmd(CONN_REFUSE, tpcb->tp_npcb, 
                    523:                                                                channel, tpcb->tp_class == TP_CLASS_4);
                    524:                                                error = EPFNOSUPPORT;
                    525:                                                goto done;
                    526:                                        }
                    527:                                 */
                    528:                                IFDEBUG(D_CONN)
                    529:                                        printf(
                    530:                                        "tp_route_to( CHANGING nlproto old 0x%x new 0x%x)\n", 
                    531:                                                        tpcb->tp_nlproto , &nl_protosw[ISO_CONS]);
                    532:                                ENDDEBUG
                    533:                                tpcb->tp_nlproto = &nl_protosw[ISO_CONS];
                    534:                        }
                    535:                        /* Now we've got the right nl_protosw. 
                    536:                         * If we already have a channel (we were called from tp_input())
                    537:                         * tell cons that we'll hang onto this channel.
                    538:                         * If we don't already have one (we were called from usrreq())
                    539:                         * -and if it's TP0 open a net connection and wait for it to finish.
                    540:                         */
                    541:                        if( channel ) {
                    542:                                error = cons_netcmd( CONN_CONFIRM, tpcb->tp_npcb, 
                    543:                                                                channel, tpcb->tp_class == TP_CLASS_4);
                    544:                                vc_to_kill ++;
                    545:                        } else if( tpcb->tp_class != TP_CLASS_4 /* class 4 only */) {
                    546:                                /* better open vc if any possibility of ending up 
                    547:                                 * in non-multiplexing class
                    548:                                 */
                    549:                                error = cons_openvc(tpcb->tp_npcb, siso, tpcb->tp_sock);
                    550:                                vc_to_kill ++;
                    551:                        }
                    552:                        /* class 4 doesn't need to open a vc now - may use one already 
                    553:                         * opened or may open one only when it sends a pkt.
                    554:                         */
                    555: #else NARGOXTWENTYFIVE > 0
                    556:                        error = ECONNREFUSED;
                    557: #endif NARGOXTWENTYFIVE > 0
                    558:                        break;
                    559:                default:
                    560:                        error = EPROTOTYPE;
                    561:                }
                    562: 
                    563:                ASSERT( save_netservice == tpcb->tp_netservice);
                    564:        }
                    565:        if( error && vc_to_kill ) {
                    566:                tp_netcmd( tpcb, CONN_CLOSE);
                    567:                goto done;
                    568:        } 
                    569:        {       /* start with the global rtt, rtv stats */
                    570:                register int i =
                    571:                   (int) tpcb->tp_flags & (TPF_PEER_ON_SAMENET | TPF_NLQOS_PDN);
                    572: 
                    573:                tpcb->tp_rtt = tp_stat.ts_rtt[i];
                    574:                tpcb->tp_rtv = tp_stat.ts_rtv[i];
                    575:        }
                    576: done:
                    577:        IFDEBUG(D_CONN)
                    578:                printf("tp_route_to  returns 0x%x\n", error);
                    579:        ENDDEBUG
                    580:        IFTRACE(D_CONN)
                    581:                tptraceTPCB(TPPTmisc, "route_to: returns: error netserv class", error, 
                    582:                        tpcb->tp_netservice, tpcb->tp_class, 0);
                    583:        ENDTRACE
                    584:        return error;
                    585: }
                    586: 
                    587: #ifdef TP_PERF_MEAS
                    588: /*
                    589:  * CALLED FROM:
                    590:  *  tp_ctloutput() when the user sets TPOPT_PERF_MEAS on
                    591:  *  and tp_newsocket() when a new connection is made from 
                    592:  *  a listening socket with tp_perf_on == true.
                    593:  * FUNCTION and ARGUMENTS:
                    594:  *  (tpcb) is the usual; this procedure gets a clear cluster mbuf for
                    595:  *  a tp_pmeas structure, and makes tpcb->tp_p_meas point to it.
                    596:  * RETURN VALUE:
                    597:  *  ENOBUFS if it cannot get a cluster mbuf.
                    598:  */
                    599: 
                    600: int 
                    601: tp_setup_perf(tpcb)
                    602:        register struct tp_pcb *tpcb;
                    603: {
                    604:        register struct mbuf *q;
                    605: 
                    606:        if( tpcb->tp_p_meas == 0 ) {
                    607:                MGET(q, M_WAITOK, MT_PCB);
                    608:                if (q == 0)
                    609:                        return ENOBUFS;
                    610:                MCLGET(q, M_WAITOK);
                    611:                if ((q->m_flags & M_EXT) == 0) {
                    612:                        (void) m_free(q);
                    613:                        return ENOBUFS;
                    614:                }
                    615:                q->m_len = sizeof (struct tp_pmeas);
                    616:                tpcb->tp_p_mbuf = q;
                    617:                tpcb->tp_p_meas = mtod(q, struct tp_pmeas *);
                    618:                bzero( (caddr_t)tpcb->tp_p_meas, sizeof (struct tp_pmeas) );
                    619:                IFDEBUG(D_PERF_MEAS)
                    620:                        printf(
                    621:                        "tpcb 0x%x so 0x%x ref 0x%x tp_p_meas 0x%x tp_perf_on 0x%x\n", 
                    622:                                tpcb, tpcb->tp_sock, tpcb->tp_lref, 
                    623:                                tpcb->tp_p_meas, tpcb->tp_perf_on);
                    624:                ENDDEBUG
                    625:                tpcb->tp_perf_on = 1;
                    626:        }
                    627:        return 0;
                    628: }
                    629: #endif TP_PERF_MEAS
                    630: 
                    631: #ifdef ARGO_DEBUG
                    632: dump_addr (addr)
                    633:        register struct sockaddr *addr;
                    634: {
                    635:        switch( addr->sa_family ) {
                    636:                case AF_INET:
                    637:                        dump_inaddr((struct sockaddr_in *)addr);
                    638:                        break;
                    639: #ifdef ISO
                    640:                case AF_ISO:
                    641:                        dump_isoaddr((struct sockaddr_iso *)addr);
                    642:                        break;
                    643: #endif ISO
                    644:                default:
                    645:                        printf("BAD AF: 0x%x\n", addr->sa_family);
                    646:                        break;
                    647:        }
                    648: }
                    649: 
                    650: #define        MAX_COLUMNS     8
                    651: /*
                    652:  *     Dump the buffer to the screen in a readable format. Format is:
                    653:  *
                    654:  *             hex/dec  where hex is the hex format, dec is the decimal format.
                    655:  *             columns of hex/dec numbers will be printed, followed by the
                    656:  *             character representations (if printable).
                    657:  */
                    658: Dump_buf(buf, len)
                    659: caddr_t        buf;
                    660: int            len;
                    661: {
                    662:        int             i,j;
                    663: 
                    664:        printf("Dump buf 0x%x len 0x%x\n", buf, len);
                    665:        for (i = 0; i < len; i += MAX_COLUMNS) {
                    666:                printf("+%d:\t", i);
                    667:                for (j = 0; j < MAX_COLUMNS; j++) {
                    668:                        if (i + j < len) {
                    669:                                printf("%x/%d\t", buf[i+j]&0xff, buf[i+j]);
                    670:                        } else {
                    671:                                printf("        ");
                    672:                        }
                    673:                }
                    674: 
                    675:                for (j = 0; j < MAX_COLUMNS; j++) {
                    676:                        if (i + j < len) {
                    677:                                if (((buf[i+j]) > 31) && ((buf[i+j]) < 128))
                    678:                                        printf("%c", buf[i+j]&0xff);
                    679:                                else
                    680:                                        printf(".");
                    681:                        }
                    682:                }
                    683:                printf("\n");
                    684:        }
                    685: }
                    686: 
                    687: 
                    688: #endif ARGO_DEBUG
                    689: 

unix.superglobalmegacorp.com

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