Annotation of XNU/bsd/netiso/tp_output.c, revision 1.1.1.1

1.1       root        1: /*
                      2:  * Copyright (c) 2000 Apple Computer, Inc. All rights reserved.
                      3:  *
                      4:  * @APPLE_LICENSE_HEADER_START@
                      5:  * 
                      6:  * The contents of this file constitute Original Code as defined in and
                      7:  * are subject to the Apple Public Source License Version 1.1 (the
                      8:  * "License").  You may not use this file except in compliance with the
                      9:  * License.  Please obtain a copy of the License at
                     10:  * http://www.apple.com/publicsource and read it before using this file.
                     11:  * 
                     12:  * This Original Code and all software distributed under the License are
                     13:  * distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER
                     14:  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
                     15:  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
                     16:  * FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT.  Please see the
                     17:  * License for the specific language governing rights and limitations
                     18:  * under the License.
                     19:  * 
                     20:  * @APPLE_LICENSE_HEADER_END@
                     21:  */
                     22: /*-
                     23:  * Copyright (c) 1991, 1993
                     24:  *     The Regents of the University of California.  All rights reserved.
                     25:  *
                     26:  * Redistribution and use in source and binary forms, with or without
                     27:  * modification, are permitted provided that the following conditions
                     28:  * are met:
                     29:  * 1. Redistributions of source code must retain the above copyright
                     30:  *    notice, this list of conditions and the following disclaimer.
                     31:  * 2. Redistributions in binary form must reproduce the above copyright
                     32:  *    notice, this list of conditions and the following disclaimer in the
                     33:  *    documentation and/or other materials provided with the distribution.
                     34:  * 3. All advertising materials mentioning features or use of this software
                     35:  *    must display the following acknowledgement:
                     36:  *     This product includes software developed by the University of
                     37:  *     California, Berkeley and its contributors.
                     38:  * 4. Neither the name of the University nor the names of its contributors
                     39:  *    may be used to endorse or promote products derived from this software
                     40:  *    without specific prior written permission.
                     41:  *
                     42:  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
                     43:  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
                     44:  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
                     45:  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
                     46:  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
                     47:  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
                     48:  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
                     49:  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
                     50:  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
                     51:  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
                     52:  * SUCH DAMAGE.
                     53:  *
                     54:  *     @(#)tp_output.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:  * In here is tp_ctloutput(), the guy called by [sg]etsockopt(),
                     87:  */
                     88: 
                     89: #include <sys/param.h>
                     90: #include <sys/mbuf.h>
                     91: #include <sys/systm.h>
                     92: #include <sys/socket.h>
                     93: #include <sys/socketvar.h>
                     94: #include <sys/protosw.h>
                     95: #include <sys/errno.h>
                     96: #include <sys/time.h>
                     97: #include <sys/kernel.h>
                     98: 
                     99: #include <netiso/tp_param.h>
                    100: #include <netiso/tp_user.h>
                    101: #include <netiso/tp_stat.h>
                    102: #include <netiso/tp_ip.h>
                    103: #include <netiso/tp_clnp.h>
                    104: #include <netiso/tp_timer.h>
                    105: #include <netiso/argo_debug.h>
                    106: #include <netiso/tp_pcb.h>
                    107: #include <netiso/tp_trace.h>
                    108: 
                    109: #define TPDUSIZESHIFT 24
                    110: #define CLASSHIFT 16
                    111: 
                    112: /*
                    113:  * NAME:       tp_consistency()
                    114:  *
                    115:  * CALLED FROM:
                    116:  *     tp_ctloutput(), tp_input()
                    117:  *
                    118:  * FUNCTION and ARGUMENTS:
                    119:  *     Checks the consistency of options and tpdusize with class,
                    120:  *     using the parameters passed in via (param).
                    121:  *     (cmd) may be TP_STRICT or TP_FORCE or both.
                    122:  *  Force means it will set all the values in (tpcb) to those in
                    123:  *  the input arguements iff no errors were encountered.
                    124:  *  Strict means that no inconsistency will be tolerated.  If it's
                    125:  *  not used, checksum and tpdusize inconsistencies will be tolerated.
                    126:  *  The reason for this is that in some cases, when we're negotiating down 
                    127:  *     from class  4, these options should be changed but should not 
                    128:  *  cause negotiation to fail.
                    129:  *
                    130:  * RETURNS
                    131:  *  E* or EOK
                    132:  *  E* if the various parms aren't ok for a given class
                    133:  *  EOK if they are ok for a given class
                    134:  */
                    135: 
                    136: int
                    137: tp_consistency( tpcb, cmd, param )
                    138:        u_int cmd;
                    139:        struct tp_conn_param *param;
                    140:        struct tp_pcb *tpcb;
                    141: {
                    142:        register int    error = EOK;
                    143:        int                     class_to_use  = tp_mask_to_num(param->p_class);
                    144: 
                    145:        IFTRACE(D_SETPARAMS)
                    146:                tptrace(TPPTmisc, 
                    147:                "tp_consist enter class_to_use dontchange param.class cmd", 
                    148:                class_to_use, param->p_dont_change_params, param->p_class, cmd);
                    149:        ENDTRACE
                    150:        IFDEBUG(D_SETPARAMS)
                    151:                printf("tp_consistency %s %s\n", 
                    152:                        cmd& TP_FORCE?  "TP_FORCE":     "",
                    153:                        cmd& TP_STRICT? "TP_STRICT":"");
                    154:        ENDDEBUG
                    155:        if ((cmd & TP_FORCE) && (param->p_dont_change_params)) {
                    156:                cmd &= ~TP_FORCE;
                    157:        }
                    158:        /* can switch net services within a domain, but
                    159:         * cannot switch domains 
                    160:         */
                    161:        switch( param->p_netservice) {
                    162:        case ISO_CONS:
                    163:        case ISO_CLNS:
                    164:        case ISO_COSNS:
                    165:                /* param->p_netservice in ISO DOMAIN */
                    166:                if(tpcb->tp_domain != AF_ISO ) {
                    167:                        error = EINVAL; goto done;
                    168:                }
                    169:                break;
                    170:        case IN_CLNS:
                    171:                /* param->p_netservice in INET DOMAIN */
                    172:                if( tpcb->tp_domain != AF_INET ) {
                    173:                        error = EINVAL; goto done;
                    174:                }
                    175:                break;
                    176:                /* no others not possible-> netservice is a 2-bit field! */
                    177:        }
                    178: 
                    179:        IFDEBUG(D_SETPARAMS)
                    180:                printf("p_class 0x%x, class_to_use 0x%x\n",  param->p_class,
                    181:                        class_to_use);
                    182:        ENDDEBUG
                    183:        if((param->p_netservice < 0) || (param->p_netservice > TP_MAX_NETSERVICES)){
                    184:                error = EINVAL; goto done;
                    185:        }
                    186:        if( (param->p_class & TP_CLASSES_IMPLEMENTED) == 0 ) {
                    187:                error = EINVAL; goto done;
                    188:        } 
                    189:        IFDEBUG(D_SETPARAMS)
                    190:                printf("Nretrans 0x%x\n",  param->p_Nretrans );
                    191:        ENDDEBUG
                    192:        if( ( param->p_Nretrans < 1 ) ||
                    193:                  (param->p_cr_ticks < 1) || (param->p_cc_ticks < 1) ) {
                    194:                        /* bad for any class because negot has to be done a la class 4 */
                    195:                        error = EINVAL; goto done;
                    196:        }
                    197:        IFDEBUG(D_SETPARAMS)
                    198:                printf("use_csum 0x%x\n",  param->p_use_checksum );
                    199:                printf("xtd_format 0x%x\n",  param->p_xtd_format );
                    200:                printf("xpd_service 0x%x\n",  param->p_xpd_service );
                    201:                printf("tpdusize 0x%x\n",  param->p_tpdusize );
                    202:                printf("tpcb->flags 0x%x\n",  tpcb->tp_flags );
                    203:        ENDDEBUG
                    204:        switch( class_to_use ) {
                    205: 
                    206:        case 0:
                    207:                /* do not use checksums, xtd format, or XPD */
                    208: 
                    209:                if( param->p_use_checksum | param->p_xtd_format | param->p_xpd_service ) {
                    210:                        if(cmd & TP_STRICT) {
                    211:                                error = EINVAL;
                    212:                        } else {
                    213:                                param->p_use_checksum = 0;
                    214:                                param->p_xtd_format = 0;
                    215:                                param->p_xpd_service = 0;
                    216:                        }
                    217:                        break;
                    218:                }
                    219: 
                    220:                if (param->p_tpdusize < TP_MIN_TPDUSIZE) {
                    221:                        if(cmd & TP_STRICT) {
                    222:                                error = EINVAL;
                    223:                        } else {
                    224:                                param->p_tpdusize = TP_MIN_TPDUSIZE;
                    225:                        }
                    226:                        break;
                    227:                }
                    228:                if (param->p_tpdusize > TP0_TPDUSIZE)  {
                    229:                        if (cmd & TP_STRICT) {
                    230:                                error = EINVAL; 
                    231:                        } else {
                    232:                                param->p_tpdusize = TP0_TPDUSIZE;
                    233:                        }
                    234:                        break;
                    235:                } 
                    236: 
                    237:                /* connect/disc data not allowed for class 0 */
                    238:                if (tpcb->tp_ucddata) {
                    239:                        if(cmd & TP_STRICT) {
                    240:                                error = EINVAL;
                    241:                        } else if(cmd & TP_FORCE) {
                    242:                                m_freem(tpcb->tp_ucddata);
                    243:                                tpcb->tp_ucddata = 0;
                    244:                        }
                    245:                }
                    246:                break;
                    247:                
                    248:        case 4:
                    249:                IFDEBUG(D_SETPARAMS)
                    250:                        printf("dt_ticks 0x%x\n",  param->p_dt_ticks );
                    251:                        printf("x_ticks 0x%x\n",  param->p_x_ticks );
                    252:                        printf("dr_ticks 0x%x\n",  param->p_dr_ticks );
                    253:                        printf("keepalive 0x%x\n",  param->p_keepalive_ticks );
                    254:                        printf("sendack 0x%x\n",  param->p_sendack_ticks );
                    255:                        printf("inact 0x%x\n",  param->p_inact_ticks );
                    256:                        printf("ref 0x%x\n",  param->p_ref_ticks );
                    257:                ENDDEBUG
                    258:                if( (param->p_class & TP_CLASS_4 ) && (
                    259:                          (param->p_dt_ticks < 1) || (param->p_dr_ticks < 1) || 
                    260:                          (param->p_x_ticks < 1)        || (param->p_keepalive_ticks < 1) ||
                    261:                          (param->p_sendack_ticks < 1) || (param->p_ref_ticks < 1) ||
                    262:                          (param->p_inact_ticks < 1) ) ) {
                    263:                                error = EINVAL;
                    264:                                break;
                    265:                }
                    266:                IFDEBUG(D_SETPARAMS)
                    267:                        printf("rx_strat 0x%x\n",  param->p_rx_strat );
                    268:                ENDDEBUG
                    269:                if(param->p_rx_strat > 
                    270:                        ( TPRX_USE_CW | TPRX_EACH | TPRX_FASTSTART) ) {
                    271:                                if(cmd & TP_STRICT) {
                    272:                                        error = EINVAL;
                    273:                                } else {
                    274:                                        param->p_rx_strat = TPRX_USE_CW;
                    275:                                }
                    276:                                break;
                    277:                }
                    278:                IFDEBUG(D_SETPARAMS)
                    279:                        printf("ack_strat 0x%x\n",  param->p_ack_strat );
                    280:                ENDDEBUG
                    281:                if((param->p_ack_strat != 0) && (param->p_ack_strat != 1)) {
                    282:                        if(cmd & TP_STRICT) {
                    283:                                error = EINVAL;
                    284:                        } else {
                    285:                                param->p_ack_strat = TPACK_WINDOW;
                    286:                        }
                    287:                        break;
                    288:                }
                    289:                if (param->p_tpdusize < TP_MIN_TPDUSIZE) {
                    290:                        if(cmd & TP_STRICT) {
                    291:                                error = EINVAL;
                    292:                        } else {
                    293:                                param->p_tpdusize = TP_MIN_TPDUSIZE;
                    294:                        }
                    295:                        break;
                    296:                }
                    297:                if (param->p_tpdusize > TP_TPDUSIZE)  {
                    298:                        if(cmd & TP_STRICT) {
                    299:                                error = EINVAL; 
                    300:                        } else {
                    301:                                param->p_tpdusize = TP_TPDUSIZE;
                    302:                        }
                    303:                        break;
                    304:                } 
                    305:                break;
                    306:        }
                    307: 
                    308:        if ((error==0) && (cmd & TP_FORCE)) {
                    309:                long dusize = ((long)param->p_ptpdusize) << 7;
                    310:                /* Enforce Negotation rules below */
                    311:                tpcb->tp_class = param->p_class;
                    312:                if (tpcb->tp_use_checksum || param->p_use_checksum)
                    313:                        tpcb->tp_use_checksum = 1;
                    314:                if (!tpcb->tp_xpd_service || !param->p_xpd_service)
                    315:                        tpcb->tp_xpd_service = 0;
                    316:                if (!tpcb->tp_xtd_format || !param->p_xtd_format)
                    317:                        tpcb->tp_xtd_format = 0;
                    318:                if (dusize) {
                    319:                        if (tpcb->tp_l_tpdusize > dusize)
                    320:                                tpcb->tp_l_tpdusize = dusize;
                    321:                        if (tpcb->tp_ptpdusize == 0 ||
                    322:                                tpcb->tp_ptpdusize > param->p_ptpdusize)
                    323:                                tpcb->tp_ptpdusize = param->p_ptpdusize;
                    324:                } else {
                    325:                        if (param->p_tpdusize != 0 &&
                    326:                                tpcb->tp_tpdusize > param->p_tpdusize)
                    327:                                tpcb->tp_tpdusize = param->p_tpdusize;
                    328:                        tpcb->tp_l_tpdusize = 1 << tpcb->tp_tpdusize;
                    329:                }
                    330:        }
                    331: done:
                    332: 
                    333:        IFTRACE(D_CONN)
                    334:                tptrace(TPPTmisc, "tp_consist returns class xtdfmt cmd", 
                    335:                        error, tpcb->tp_class, tpcb->tp_xtd_format, cmd);
                    336:        ENDTRACE
                    337:        IFDEBUG(D_CONN)
                    338:                printf(
                    339:                "tp_consist rtns 0x%x class 0x%x xtd_fmt 0x%x cmd 0x%x\n",
                    340:                        error, tpcb->tp_class, tpcb->tp_xtd_format, cmd);
                    341:        ENDDEBUG
                    342:        return error;
                    343: }
                    344: 
                    345: /*
                    346:  * NAME:       tp_ctloutput()
                    347:  *
                    348:  * CALLED FROM:
                    349:  *     [sg]etsockopt(), via so[sg]etopt(). 
                    350:  *
                    351:  * FUNCTION and ARGUMENTS:
                    352:  *     Implements the socket options at transport level.
                    353:  *     (cmd) is either PRCO_SETOPT or PRCO_GETOPT (see ../sys/protosw.h).
                    354:  *     (so) is the socket.
                    355:  *     (level) is SOL_TRANSPORT (see ../sys/socket.h)
                    356:  *     (optname) is the particular command or option to be set.
                    357:  *     (**mp) is an mbuf structure.  
                    358:  *
                    359:  * RETURN VALUE:
                    360:  *     ENOTSOCK if the socket hasn't got an associated tpcb
                    361:  *  EINVAL if 
                    362:  *             trying to set window too big
                    363:  *             trying to set illegal max tpdu size 
                    364:  *             trying to set illegal credit fraction
                    365:  *             trying to use unknown or unimplemented class of TP
                    366:  *             structure passed to set timer values is wrong size
                    367:  *     illegal combination of command/GET-SET option, 
                    368:  *                     e.g., GET w/ TPOPT_CDDATA_CLEAR: 
                    369:  *  EOPNOTSUPP if the level isn't transport, or command is neither GET nor SET
                    370:  *   or if the transport-specific command is not implemented
                    371:  *  EISCONN if trying a command that isn't allowed after a connection
                    372:  *   is established
                    373:  *  ENOTCONN if trying a command that is allowed only if a connection is
                    374:  *   established
                    375:  *  EMSGSIZE if trying to give too much data on connect/disconnect
                    376:  *
                    377:  * SIDE EFFECTS:
                    378:  *
                    379:  * NOTES:
                    380:  */
                    381: ProtoHook 
                    382: tp_ctloutput(cmd, so, level, optname, mp)
                    383:        int                     cmd, level, optname;
                    384:        struct socket   *so;
                    385:        struct mbuf     **mp;
                    386: {
                    387:        struct          tp_pcb  *tpcb = sototpcb(so);
                    388:        int             s = splnet();
                    389:        caddr_t         value;
                    390:        unsigned        val_len;
                    391:        int                     error = 0;
                    392: 
                    393:        IFTRACE(D_REQUEST)
                    394:                tptrace(TPPTmisc, "tp_ctloutput cmd so optname mp", 
                    395:                        cmd, so, optname, mp);
                    396:        ENDTRACE
                    397:        IFDEBUG(D_REQUEST)
                    398:                printf(
                    399:        "tp_ctloutput so 0x%x cmd 0x%x optname 0x%x, mp 0x%x *mp 0x%x tpcb 0x%x\n", 
                    400:                        so, cmd, optname, mp, mp?*mp:0, tpcb);
                    401:        ENDDEBUG
                    402:        if( tpcb == (struct tp_pcb *)0 ) {
                    403:                error = ENOTSOCK; goto done;
                    404:        }
                    405:        if(*mp == MNULL) {
                    406:                register struct mbuf *m;
                    407: 
                    408:                MGET(m, M_DONTWAIT, TPMT_SONAME); /* does off, type, next */
                    409:                if (m == NULL) {
                    410:                        splx(s);
                    411:                        return ENOBUFS;
                    412:                }
                    413:                m->m_len = 0;
                    414:                m->m_act = 0;
                    415:                *mp = m;
                    416:        }
                    417: 
                    418:        /*
                    419:         *      Hook so one can set network options via a tp socket.
                    420:         */
                    421:        if ( level == SOL_NETWORK ) {
                    422:                if ((tpcb->tp_nlproto == NULL) || (tpcb->tp_npcb == NULL))
                    423:                        error = ENOTSOCK;
                    424:                else if (tpcb->tp_nlproto->nlp_ctloutput == NULL)
                    425:                        error = EOPNOTSUPP;
                    426:                else
                    427:                        return ((tpcb->tp_nlproto->nlp_ctloutput)(cmd, optname, 
                    428:                                tpcb->tp_npcb, *mp));
                    429:                goto done;
                    430:        } else if ( level == SOL_SOCKET) {
                    431:                if (optname == SO_RCVBUF && cmd == PRCO_SETOPT) {
                    432:                        u_long old_credit = tpcb->tp_maxlcredit;
                    433:                        tp_rsyset(tpcb);
                    434:                        if (tpcb->tp_rhiwat != so->so_rcv.sb_hiwat &&
                    435:                            tpcb->tp_state == TP_OPEN &&
                    436:                            (old_credit < tpcb->tp_maxlcredit))
                    437:                                tp_emit(AK_TPDU_type, tpcb,
                    438:                                        tpcb->tp_rcvnxt, 0, MNULL);
                    439:                        tpcb->tp_rhiwat = so->so_rcv.sb_hiwat;
                    440:                }
                    441:                goto done;
                    442:        } else if ( level !=  SOL_TRANSPORT ) {
                    443:                error = EOPNOTSUPP; goto done;
                    444:        } 
                    445:        if (cmd != PRCO_GETOPT && cmd != PRCO_SETOPT) {
                    446:                error = EOPNOTSUPP; goto done;
                    447:        } 
                    448:        if ( so->so_error ) {
                    449:                error = so->so_error; goto done;
                    450:        }
                    451: 
                    452:        /* The only options allowed after connection is established
                    453:         * are GET (anything) and SET DISC DATA and SET PERF MEAS
                    454:         */
                    455:        if ( ((so->so_state & SS_ISCONNECTING)||(so->so_state & SS_ISCONNECTED))
                    456:                &&
                    457:                (cmd == PRCO_SETOPT  && 
                    458:                        optname != TPOPT_DISC_DATA && 
                    459:                        optname != TPOPT_CFRM_DATA && 
                    460:                        optname != TPOPT_PERF_MEAS &&
                    461:                        optname != TPOPT_CDDATA_CLEAR ) ) {
                    462:                error = EISCONN; goto done;
                    463:        } 
                    464:        /* The only options allowed after disconnection are GET DISC DATA,
                    465:         * and TPOPT_PSTATISTICS
                    466:         * and they're not allowed if the ref timer has gone off, because
                    467:         * the tpcb is gone 
                    468:         */
                    469:        if ((so->so_state & (SS_ISCONNECTED | SS_ISCONFIRMING)) ==  0) {
                    470:                if ( so->so_pcb == (caddr_t)0 ) {
                    471:                        error = ENOTCONN; goto done;
                    472:                }
                    473:                if ( (tpcb->tp_state == TP_REFWAIT || tpcb->tp_state == TP_CLOSING) &&
                    474:                                (optname != TPOPT_DISC_DATA && optname != TPOPT_PSTATISTICS)) {
                    475:                        error = ENOTCONN; goto done;
                    476:                }
                    477:        }
                    478: 
                    479:        value = mtod(*mp, caddr_t);  /* it's aligned, don't worry,
                    480:                                                                  * but lint complains about it 
                    481:                                                                  */
                    482:        val_len = (*mp)->m_len;
                    483: 
                    484:        switch (optname) {
                    485: 
                    486:        case TPOPT_INTERCEPT:
                    487: #define INA(t) (((struct inpcb *)(t->tp_npcb))->inp_laddr.s_addr)
                    488: #define ISOA(t) (((struct isopcb *)(t->tp_npcb))->isop_laddr->siso_addr)
                    489: 
                    490:                if ((so->so_state & SS_PRIV) == 0) {
                    491:                        error = EPERM;
                    492:                } else if (cmd != PRCO_SETOPT || tpcb->tp_state != TP_CLOSED ||
                    493:                                        (tpcb->tp_flags & TPF_GENERAL_ADDR) ||
                    494:                                        tpcb->tp_next == 0)
                    495:                        error = EINVAL;
                    496:                else {
                    497:                        register struct tp_pcb *t;
                    498:                        error = EADDRINUSE;
                    499:                        for (t = tp_listeners; t; t = t->tp_nextlisten)
                    500:                                if ((t->tp_flags & TPF_GENERAL_ADDR) == 0 &&
                    501:                                                t->tp_domain == tpcb->tp_domain)
                    502:                                        switch (tpcb->tp_domain) {
                    503:                                        default:
                    504:                                                goto done;
                    505: #if    INET
                    506:                                        case AF_INET:
                    507:                                                if (INA(t) == INA(tpcb))
                    508:                                                        goto done;
                    509:                                                continue;
                    510: #endif
                    511: #if ISO
                    512:                                        case AF_ISO:
                    513:                                                if (bcmp(ISOA(t).isoa_genaddr, ISOA(tpcb).isoa_genaddr,
                    514:                                                                                ISOA(t).isoa_len) == 0)
                    515:                                                        goto done;
                    516:                                                continue;
                    517: #endif
                    518:                                        }
                    519:                        tpcb->tp_lsuffixlen = 0;
                    520:                        tpcb->tp_state = TP_LISTENING;
                    521:                        error = 0;
                    522:                        remque(tpcb);
                    523:                        tpcb->tp_next = tpcb->tp_prev = tpcb;
                    524:                        tpcb->tp_nextlisten = tp_listeners;
                    525:                        tp_listeners = tpcb;
                    526:                }
                    527:                break;
                    528: 
                    529:        case TPOPT_MY_TSEL:
                    530:                if ( cmd == PRCO_GETOPT ) {
                    531:                        ASSERT( tpcb->tp_lsuffixlen <= MAX_TSAP_SEL_LEN );
                    532:                        bcopy((caddr_t)tpcb->tp_lsuffix, value, tpcb->tp_lsuffixlen);
                    533:                        (*mp)->m_len = tpcb->tp_lsuffixlen;
                    534:                } else /* cmd == PRCO_SETOPT  */ {
                    535:                        if( (val_len > MAX_TSAP_SEL_LEN) || (val_len <= 0 )) {
                    536:                                printf("val_len 0x%x (*mp)->m_len 0x%x\n", val_len, (*mp));
                    537:                                error = EINVAL;
                    538:                        } else {
                    539:                                bcopy(value, (caddr_t)tpcb->tp_lsuffix, val_len);
                    540:                                tpcb->tp_lsuffixlen = val_len;
                    541:                        }
                    542:                }
                    543:                break;
                    544: 
                    545:        case TPOPT_PEER_TSEL:
                    546:                if ( cmd == PRCO_GETOPT ) {
                    547:                        ASSERT( tpcb->tp_fsuffixlen <= MAX_TSAP_SEL_LEN );
                    548:                        bcopy((caddr_t)tpcb->tp_fsuffix, value, tpcb->tp_fsuffixlen);
                    549:                        (*mp)->m_len = tpcb->tp_fsuffixlen;
                    550:                } else /* cmd == PRCO_SETOPT  */ {
                    551:                        if( (val_len > MAX_TSAP_SEL_LEN) || (val_len <= 0 )) {
                    552:                                printf("val_len 0x%x (*mp)->m_len 0x%x\n", val_len, (*mp));
                    553:                                error = EINVAL; 
                    554:                        } else {
                    555:                                bcopy(value, (caddr_t)tpcb->tp_fsuffix, val_len);
                    556:                                tpcb->tp_fsuffixlen = val_len;
                    557:                        }
                    558:                }
                    559:                break;
                    560: 
                    561:        case TPOPT_FLAGS:
                    562:                IFDEBUG(D_REQUEST)
                    563:                        printf("%s TPOPT_FLAGS value 0x%x *value 0x%x, flags 0x%x \n", 
                    564:                                cmd==PRCO_GETOPT?"GET":"SET", 
                    565:                                value,
                    566:                                *value, 
                    567:                                tpcb->tp_flags);
                    568:                ENDDEBUG
                    569: 
                    570:                if ( cmd == PRCO_GETOPT ) {
                    571:                        *(int *)value = (int)tpcb->tp_flags;
                    572:                        (*mp)->m_len = sizeof(u_int);
                    573:                } else /* cmd == PRCO_SETOPT  */ {
                    574:                        error = EINVAL; goto done;
                    575:                }
                    576:                break;
                    577: 
                    578:        case TPOPT_PARAMS:
                    579:                /* This handles:
                    580:                 * timer values,
                    581:                 * class, use of transport expedited data,
                    582:                 * max tpdu size, checksum, xtd format and
                    583:                 * disconnect indications, and may get rid of connect/disc data
                    584:                 */
                    585:                IFDEBUG(D_SETPARAMS)
                    586:                        printf("TPOPT_PARAMS value 0x%x, cmd %s \n", value,
                    587:                                cmd==PRCO_GETOPT?"GET":"SET");
                    588:                ENDDEBUG
                    589:                IFDEBUG(D_REQUEST)
                    590:                        printf("TPOPT_PARAMS value 0x%x, cmd %s \n", value,
                    591:                                cmd==PRCO_GETOPT?"GET":"SET");
                    592:                ENDDEBUG
                    593: 
                    594:                if ( cmd == PRCO_GETOPT ) {
                    595:                        *(struct tp_conn_param *)value = tpcb->_tp_param;
                    596:                        (*mp)->m_len = sizeof(tpcb->_tp_param);
                    597:                } else /* cmd == PRCO_SETOPT  */ {
                    598:                        if( (error = 
                    599:                                tp_consistency(tpcb, TP_STRICT | TP_FORCE, 
                    600:                                                                (struct tp_conn_param *)value))==0) {
                    601:                                /* 
                    602:                                 * tp_consistency doesn't copy the whole set of params 
                    603:                                 */
                    604:                                tpcb->_tp_param = *(struct tp_conn_param *)value;
                    605:                                (*mp)->m_len = sizeof(tpcb->_tp_param);
                    606:                        }
                    607:                }
                    608:                break;
                    609: 
                    610:        case TPOPT_PSTATISTICS: 
                    611: #ifdef TP_PERF_MEAS
                    612:                if (cmd == PRCO_SETOPT) {
                    613:                        error = EINVAL; goto done;
                    614:                } 
                    615:                IFPERF(tpcb)
                    616:                        if (*mp) {
                    617:                                struct mbuf * n;
                    618:                                do {
                    619:                                        MFREE(*mp, n);
                    620:                                        *mp = n;
                    621:                                } while (n);
                    622:                        }
                    623:                        *mp = m_copym(tpcb->tp_p_mbuf, (int)M_COPYALL, M_WAITOK);
                    624:                ENDPERF 
                    625:                else {
                    626:                        error = EINVAL; goto done;
                    627:                } 
                    628:                break;
                    629: #else
                    630:                error = EOPNOTSUPP;
                    631:                goto done;
                    632: #endif /* TP_PERF_MEAS */
                    633:                
                    634:        case TPOPT_CDDATA_CLEAR: 
                    635:                if (cmd == PRCO_GETOPT) {
                    636:                        error = EINVAL;
                    637:                } else {
                    638:                        if (tpcb->tp_ucddata) {
                    639:                                m_freem(tpcb->tp_ucddata);
                    640:                                tpcb->tp_ucddata = 0;
                    641:                        }
                    642:                }
                    643:                break;
                    644: 
                    645:        case TPOPT_CFRM_DATA:
                    646:        case TPOPT_DISC_DATA: 
                    647:        case TPOPT_CONN_DATA: 
                    648:                if( tpcb->tp_class == TP_CLASS_0 ) {
                    649:                        error = EOPNOTSUPP;
                    650:                        break;
                    651:                }
                    652:                IFDEBUG(D_REQUEST)
                    653:                        printf("%s\n", optname==TPOPT_DISC_DATA?"DISC data":"CONN data");
                    654:                        printf("m_len 0x%x, vallen 0x%x so_snd.cc 0x%x\n", 
                    655:                                (*mp)->m_len, val_len, so->so_snd.sb_cc);
                    656:                        dump_mbuf(so->so_snd.sb_mb, "tp_ctloutput: sosnd ");
                    657:                ENDDEBUG
                    658:                if (cmd == PRCO_SETOPT) {
                    659:                        int len = tpcb->tp_ucddata ?  tpcb->tp_ucddata->m_len : 0;
                    660:                        /* can append connect data in several calls */
                    661:                        if (len + val_len > 
                    662:                                (optname==TPOPT_CONN_DATA?TP_MAX_CR_DATA:TP_MAX_DR_DATA) ) {
                    663:                                error = EMSGSIZE; goto done;
                    664:                        } 
                    665:                        (*mp)->m_next = MNULL;
                    666:                        (*mp)->m_act = 0;
                    667:                        if (tpcb->tp_ucddata)
                    668:                                m_cat(tpcb->tp_ucddata, *mp);
                    669:                        else
                    670:                                tpcb->tp_ucddata = *mp;
                    671:                        IFDEBUG(D_REQUEST)
                    672:                                dump_mbuf(tpcb->tp_ucddata, "tp_ctloutput after CONN_DATA");
                    673:                        ENDDEBUG
                    674:                        IFTRACE(D_REQUEST)
                    675:                                tptrace(TPPTmisc,"C/D DATA: flags snd.sbcc val_len",
                    676:                                        tpcb->tp_flags, so->so_snd.sb_cc,val_len,0);
                    677:                        ENDTRACE
                    678:                        *mp = MNULL;
                    679:                        if (optname == TPOPT_CFRM_DATA && (so->so_state & SS_ISCONFIRMING))
                    680:                                (void) tp_confirm(tpcb);
                    681:                }
                    682:                break;
                    683: 
                    684:        case TPOPT_PERF_MEAS: 
                    685: #ifdef TP_PERF_MEAS
                    686:                if (cmd == PRCO_GETOPT) {
                    687:                        *value = (u_int)tpcb->tp_perf_on;
                    688:                        (*mp)->m_len = sizeof(u_int);
                    689:                } else if (cmd == PRCO_SETOPT) {
                    690:                        (*mp)->m_len = 0;
                    691:                        if ((*value) != 0 && (*value) != 1 )
                    692:                                error = EINVAL;
                    693:                        else  tpcb->tp_perf_on = (*value);
                    694:                }
                    695:                if( tpcb->tp_perf_on ) 
                    696:                        error = tp_setup_perf(tpcb);
                    697: #else  /* TP_PERF_MEAS */
                    698:                error = EOPNOTSUPP;
                    699: #endif /* TP_PERF_MEAS */
                    700:                break;
                    701: 
                    702:        default:
                    703:                error = EOPNOTSUPP;
                    704:        }
                    705:        
                    706: done:
                    707:        IFDEBUG(D_REQUEST)
                    708:                dump_mbuf(so->so_snd.sb_mb, "tp_ctloutput sosnd at end");
                    709:                dump_mbuf(*mp, "tp_ctloutput *mp");
                    710:        ENDDEBUG
                    711:        /* 
                    712:         * sigh: getsockopt looks only at m_len : all output data must 
                    713:         * reside in the first mbuf 
                    714:         */
                    715:        if (*mp) {
                    716:                if (cmd == PRCO_SETOPT) {
                    717:                        m_freem(*mp);
                    718:                        *mp = MNULL;
                    719:                } else {
                    720:                        ASSERT ( m_compress(*mp, mp) <= MLEN );
                    721:                        if (error)
                    722:                                (*mp)->m_len = 0;
                    723:                        IFDEBUG(D_REQUEST)
                    724:                                dump_mbuf(*mp, "tp_ctloutput *mp after compress");
                    725:                        ENDDEBUG
                    726:                }
                    727:        }
                    728:        splx(s);
                    729:        return error;
                    730: }

unix.superglobalmegacorp.com

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