Annotation of XNU/bsd/netccitt/hd_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) University of British Columbia, 1984
                     24:  * Copyright (c) 1990, 1993
                     25:  *     The Regents of the University of California.  All rights reserved.
                     26:  *
                     27:  * This code is derived from software contributed to Berkeley by
                     28:  * the Laboratory for Computation Vision and the Computer Science Department
                     29:  * of the University of British Columbia.
                     30:  *
                     31:  * Redistribution and use in source and binary forms, with or without
                     32:  * modification, are permitted provided that the following conditions
                     33:  * are met:
                     34:  * 1. Redistributions of source code must retain the above copyright
                     35:  *    notice, this list of conditions and the following disclaimer.
                     36:  * 2. Redistributions in binary form must reproduce the above copyright
                     37:  *    notice, this list of conditions and the following disclaimer in the
                     38:  *    documentation and/or other materials provided with the distribution.
                     39:  * 3. All advertising materials mentioning features or use of this software
                     40:  *    must display the following acknowledgement:
                     41:  *     This product includes software developed by the University of
                     42:  *     California, Berkeley and its contributors.
                     43:  * 4. Neither the name of the University nor the names of its contributors
                     44:  *    may be used to endorse or promote products derived from this software
                     45:  *    without specific prior written permission.
                     46:  *
                     47:  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
                     48:  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
                     49:  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
                     50:  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
                     51:  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
                     52:  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
                     53:  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
                     54:  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
                     55:  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
                     56:  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
                     57:  * SUCH DAMAGE.
                     58:  *
                     59:  *     @(#)hd_output.c 8.1 (Berkeley) 6/10/93
                     60:  */
                     61: 
                     62: #include <sys/param.h>
                     63: #include <sys/systm.h>
                     64: #include <sys/mbuf.h>
                     65: #include <sys/domain.h>
                     66: #include <sys/socket.h>
                     67: #include <sys/syslog.h>
                     68: #include <sys/protosw.h>
                     69: #include <sys/errno.h>
                     70: #include <sys/time.h>
                     71: #include <sys/kernel.h>
                     72: 
                     73: #include <net/if.h>
                     74: 
                     75: #include <netccitt/hdlc.h>
                     76: #include <netccitt/hd_var.h>
                     77: #include <netccitt/x25.h>
                     78: 
                     79: /*
                     80:  *      HDLC OUTPUT INTERFACE
                     81:  *
                     82:  *      This routine is called when the X.25 packet layer output routine
                     83:  *      has a information frame (iframe)  to write.   It is  also called 
                     84:  *      by the input and control routines of the HDLC layer.
                     85:  */
                     86: 
                     87: hd_output (hdp, m0)
                     88: register struct hdcb *hdp;
                     89: struct mbuf *m0;
                     90: {
                     91:        struct x25config *xcp;
                     92:        register struct mbuf *m = m0;
                     93:        int len;
                     94: 
                     95:        if (m == NULL)
                     96:                panic ("hd_output");
                     97:        if ((m->m_flags & M_PKTHDR) == 0)
                     98:                panic ("hd_output 2");
                     99: 
                    100:        if (hdp->hd_state != ABM) {
                    101:                m_freem (m);
                    102:                return;
                    103:        }
                    104: 
                    105:        /*
                    106:         * Make room for the hdlc header either by prepending
                    107:         * another mbuf, or by adjusting the offset and length
                    108:         * of the first mbuf in the mbuf chain.
                    109:         */
                    110: 
                    111:        M_PREPEND(m, HDHEADERLN, M_DONTWAIT);
                    112:        if (m == NULL)
                    113:                return;
                    114:        for (len = 0; m; m = m->m_next)
                    115:                len += m->m_len;
                    116:        m = m0;
                    117:        m->m_pkthdr.len = len;
                    118: 
                    119:        hd_append (&hdp->hd_txq, m);
                    120:        hd_start (hdp);
                    121: }
                    122: 
                    123: hd_start (hdp)
                    124: register struct hdcb *hdp;
                    125: {
                    126:        register struct mbuf *m;
                    127: 
                    128:        /* 
                    129:         * The iframe is only transmitted if all these conditions are FALSE.
                    130:         * The iframe remains queued (hdp->hd_txq) however and will be
                    131:         * transmitted as soon as these conditions are cleared.
                    132:         */
                    133: 
                    134:        while (!(hdp->hd_condition & (TIMER_RECOVERY_CONDITION | REMOTE_RNR_CONDITION | REJ_CONDITION))) {
                    135:                if (hdp->hd_vs == (hdp->hd_lastrxnr + hdp->hd_xcp->xc_lwsize) % MODULUS) {
                    136: 
                    137:                        /* We have now exceeded the  maximum  number  of 
                    138:                           outstanding iframes. Therefore,  we must wait 
                    139:                           until  at least  one is acknowledged if this 
                    140:                           condition  is not  turned off before we are
                    141:                           requested to write another iframe. */
                    142:                        hdp->hd_window_condition++;
                    143:                        break;
                    144:                }
                    145: 
                    146:                /* hd_remove top iframe from transmit queue. */
                    147:                if ((m = hd_remove (&hdp->hd_txq)) == NULL)
                    148:                        break;
                    149: 
                    150:                hd_send_iframe (hdp, m, POLLOFF);
                    151:        }
                    152: }
                    153: 
                    154: /* 
                    155:  *  This procedure is passed a buffer descriptor for an iframe. It builds
                    156:  *  the rest of the control part of the frame and then writes it out.  It
                    157:  *  also  starts the  acknowledgement  timer and keeps  the iframe in the
                    158:  *  Retransmit queue (Retxq) just in case  we have to do this again.
                    159:  *
                    160:  *  Note: This routine is also called from hd_input.c when retransmission
                    161:  *       of old frames is required.
                    162:  */
                    163: 
                    164: hd_send_iframe (hdp, buf, poll_bit)
                    165: register struct hdcb *hdp;
                    166: register struct mbuf *buf;
                    167: int poll_bit;
                    168: {
                    169:        register struct Hdlc_iframe *iframe;
                    170:        struct mbuf *m;
                    171: 
                    172:        KILL_TIMER (hdp);
                    173: 
                    174:        if (buf == 0) {
                    175:                printf ("hd_send_iframe: zero arg\n");
                    176: #ifdef HDLCDEBUG
                    177:                hd_status (hdp);
                    178:                hd_dumptrace (hdp);
                    179: #endif
                    180:                hdp->hd_vs = (hdp->hd_vs + 7) % MODULUS;
                    181:                return;
                    182:        }
                    183:        iframe = mtod (buf, struct Hdlc_iframe *);
                    184: 
                    185:        iframe -> hdlc_0 = 0;
                    186:        iframe -> nr = hdp->hd_vr;
                    187:        iframe -> pf = poll_bit;
                    188:        iframe -> ns = hdp->hd_vs;
                    189:        iframe -> address = ADDRESS_B;
                    190:        hdp->hd_lasttxnr = hdp->hd_vr;
                    191:        hdp->hd_rrtimer = 0;
                    192: 
                    193:        if (hdp->hd_vs == hdp->hd_retxqi) {
                    194:                /* Check for retransmissions. */
                    195:                /* Put iframe only once in the Retransmission queue. */
                    196:                hdp->hd_retxq[hdp->hd_retxqi] = buf;
                    197:                hdp->hd_retxqi = (hdp->hd_retxqi + 1) % MODULUS;
                    198:                hdp->hd_iframes_out++;
                    199:        }
                    200: 
                    201:        hdp->hd_vs = (hdp->hd_vs + 1) % MODULUS;
                    202: 
                    203:        hd_trace (hdp, TX, (struct Hdlc_frame *)iframe);
                    204: 
                    205:        /* Write buffer on device. */
                    206:        m = hdp->hd_dontcopy ? buf : m_copy(buf, 0, (int)M_COPYALL);
                    207:        if (m == 0) {
                    208:                printf("hdlc: out of mbufs\n");
                    209:                return;
                    210:        }
                    211:        (*hdp->hd_output)(hdp, m);
                    212:        SET_TIMER (hdp);
                    213: }
                    214: 
                    215: hd_ifoutput(hdp, m)
                    216: register struct mbuf *m;
                    217: register struct hdcb *hdp;
                    218: {
                    219:        /*
                    220:         * Queue message on interface, and start output if interface
                    221:         * not yet active.
                    222:         */
                    223:        register struct ifnet *ifp = hdp->hd_ifp;
                    224:        int s = splimp();
                    225: 
                    226:        if (IF_QFULL(&ifp->if_snd)) {
                    227:                IF_DROP(&ifp->if_snd);
                    228:            /* printf("%s%d: HDLC says OK to send but queue full, may hang\n",
                    229:                        ifp->if_name, ifp->if_unit);*/
                    230:                m_freem(m);
                    231:        } else {
                    232:                IF_ENQUEUE(&ifp->if_snd, m);
                    233:                if ((ifp->if_flags & IFF_OACTIVE) == 0)
                    234:                        (*ifp->if_start)(ifp);
                    235:        }
                    236:        splx(s);
                    237: }
                    238: 
                    239: 
                    240: /* 
                    241:  *  This routine gets control when the timer expires because we have not
                    242:  *  received an acknowledgement for a iframe.
                    243:  */
                    244: 
                    245: hd_resend_iframe (hdp)
                    246: register struct hdcb *hdp;
                    247: {
                    248: 
                    249:        if (hdp->hd_retxcnt++ < hd_n2) {
                    250:                if (!(hdp->hd_condition & TIMER_RECOVERY_CONDITION)) {
                    251:                        hdp->hd_xx = hdp->hd_vs;
                    252:                        hdp->hd_condition |= TIMER_RECOVERY_CONDITION;
                    253:                }
                    254: 
                    255:                hdp->hd_vs = hdp->hd_lastrxnr;
                    256:                hd_send_iframe (hdp, hdp->hd_retxq[hdp->hd_vs], POLLON);
                    257:        } else {
                    258:                /* At this point we have not received a RR even after N2
                    259:                   retries - attempt to reset link. */
                    260: 
                    261:                hd_initvars (hdp);
                    262:                hd_writeinternal (hdp, SABM, POLLOFF);
                    263:                hdp->hd_state = WAIT_UA;
                    264:                SET_TIMER (hdp);
                    265:                hd_message (hdp, "Timer recovery failed: link down");
                    266:                (void) pk_ctlinput (PRC_LINKDOWN, hdp->hd_pkp);
                    267:        }
                    268: }

unix.superglobalmegacorp.com

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