|
|
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: * @(#)clnp_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: #include <sys/param.h> ! 85: #include <sys/mbuf.h> ! 86: #include <sys/domain.h> ! 87: #include <sys/protosw.h> ! 88: #include <sys/socket.h> ! 89: #include <sys/socketvar.h> ! 90: #include <sys/errno.h> ! 91: #include <sys/time.h> ! 92: ! 93: #include <net/if.h> ! 94: #include <net/route.h> ! 95: ! 96: #include <netiso/iso.h> ! 97: #include <netiso/iso_var.h> ! 98: #include <netiso/iso_pcb.h> ! 99: #include <netiso/clnp.h> ! 100: #include <netiso/clnp_stat.h> ! 101: #include <netiso/argo_debug.h> ! 102: ! 103: static struct clnp_fixed dt_template = { ! 104: ISO8473_CLNP, /* network identifier */ ! 105: 0, /* length */ ! 106: ISO8473_V1, /* version */ ! 107: CLNP_TTL, /* ttl */ ! 108: CLNP_DT|CNF_SEG_OK|CNF_ERR_OK, /* type */ ! 109: 0, /* segment length */ ! 110: 0 /* checksum */ ! 111: }; ! 112: ! 113: static struct clnp_fixed raw_template = { ! 114: ISO8473_CLNP, /* network identifier */ ! 115: 0, /* length */ ! 116: ISO8473_V1, /* version */ ! 117: CLNP_TTL, /* ttl */ ! 118: CLNP_RAW|CNF_SEG_OK|CNF_ERR_OK, /* type */ ! 119: 0, /* segment length */ ! 120: 0 /* checksum */ ! 121: }; ! 122: ! 123: static struct clnp_fixed echo_template = { ! 124: ISO8473_CLNP, /* network identifier */ ! 125: 0, /* length */ ! 126: ISO8473_V1, /* version */ ! 127: CLNP_TTL, /* ttl */ ! 128: CLNP_EC|CNF_SEG_OK|CNF_ERR_OK, /* type */ ! 129: 0, /* segment length */ ! 130: 0 /* checksum */ ! 131: }; ! 132: ! 133: static struct clnp_fixed echor_template = { ! 134: ISO8473_CLNP, /* network identifier */ ! 135: 0, /* length */ ! 136: ISO8473_V1, /* version */ ! 137: CLNP_TTL, /* ttl */ ! 138: CLNP_ECR|CNF_SEG_OK|CNF_ERR_OK, /* type */ ! 139: 0, /* segment length */ ! 140: 0 /* checksum */ ! 141: }; ! 142: ! 143: #ifdef DECBIT ! 144: u_char qos_option[] = {CLNPOVAL_QOS, 1, ! 145: CLNPOVAL_GLOBAL|CLNPOVAL_SEQUENCING|CLNPOVAL_LOWDELAY}; ! 146: #endif /* DECBIT */ ! 147: ! 148: int clnp_id = 0; /* id for segmented dgrams */ ! 149: ! 150: /* ! 151: * FUNCTION: clnp_output ! 152: * ! 153: * PURPOSE: output the data in the mbuf as a clnp datagram ! 154: * ! 155: * The data specified by m0 is sent as a clnp datagram. ! 156: * The mbuf chain m0 will be freed when this routine has ! 157: * returned. ! 158: * ! 159: * If options is non-null, it points to an mbuf which contains ! 160: * options to be sent with the datagram. The options must ! 161: * be formatted in the mbuf according to clnp rules. Options ! 162: * will not be freed. ! 163: * ! 164: * Datalen specifies the length of the data in m0. ! 165: * ! 166: * Src and dst are the addresses for the packet. ! 167: * ! 168: * If route is non-null, it is used as the route for ! 169: * the packet. ! 170: * ! 171: * By default, a DT is sent. However, if flags & CNLP_SEND_ER ! 172: * then an ER will be sent. If flags & CLNP_SEND_RAW, then ! 173: * the packet will be send as raw clnp. ! 174: * ! 175: * RETURNS: 0 success ! 176: * appropriate error code ! 177: * ! 178: * SIDE EFFECTS: none ! 179: * ! 180: * NOTES: ! 181: * Flags are interpretated as follows: ! 182: * CLNP_NO_SEG - do not allow this pkt to be segmented. ! 183: * CLNP_NO_ER - have pkt request ER suppression. ! 184: * CLNP_SEND_RAW - send pkt as RAW DT rather than TP DT ! 185: * CLNP_NO_CKSUM - don't compute clnp checksum ! 186: * CLNP_ECHO - send as ECHO packet ! 187: * ! 188: * When checking for a cached packet, clnp checks ! 189: * that the route taken is still up. It does not ! 190: * check that the route is still to the same destination. ! 191: * This means that any entity that alters an existing ! 192: * route for an isopcb (such as when a redirect arrives) ! 193: * must invalidate the clnp cache. It might be perferable ! 194: * to have clnp check that the route has the same dest, but ! 195: * by avoiding this check, we save a call to iso_addrmatch1. ! 196: */ ! 197: clnp_output(m0, isop, datalen, flags) ! 198: struct mbuf *m0; /* data for the packet */ ! 199: struct isopcb *isop; /* iso pcb */ ! 200: int datalen; /* number of bytes of data in m0 */ ! 201: int flags; /* flags */ ! 202: { ! 203: int error = 0; /* return value of function */ ! 204: register struct mbuf *m = m0; /* mbuf for clnp header chain */ ! 205: register struct clnp_fixed *clnp; /* ptr to fixed part of hdr */ ! 206: register caddr_t hoff; /* offset into header */ ! 207: int total_len; /* total length of packet */ ! 208: struct iso_addr *src; /* ptr to source address */ ! 209: struct iso_addr *dst; /* ptr to destination address */ ! 210: struct clnp_cache clc; /* storage for cache information */ ! 211: struct clnp_cache *clcp = NULL; /* ptr to clc */ ! 212: int hdrlen = 0; ! 213: ! 214: dst = &isop->isop_faddr->siso_addr; ! 215: if (isop->isop_laddr == 0) { ! 216: struct iso_ifaddr *ia = 0; ! 217: clnp_route(dst, &isop->isop_route, flags, 0, &ia); ! 218: if (ia == 0 || ia->ia_ifa.ifa_addr->sa_family != AF_ISO) ! 219: return (ENETUNREACH); ! 220: src = &ia->ia_addr.siso_addr; ! 221: } else ! 222: src = &isop->isop_laddr->siso_addr; ! 223: ! 224: IFDEBUG(D_OUTPUT) ! 225: printf("clnp_output: to %s", clnp_iso_addrp(dst)); ! 226: printf(" from %s of %d bytes\n", clnp_iso_addrp(src), datalen); ! 227: printf("\toptions x%x, flags x%x, isop_clnpcache x%x\n", ! 228: isop->isop_options, flags, isop->isop_clnpcache); ! 229: ENDDEBUG ! 230: ! 231: if (isop->isop_clnpcache != NULL) { ! 232: clcp = mtod(isop->isop_clnpcache, struct clnp_cache *); ! 233: } ! 234: ! 235: /* ! 236: * Check if cache is valid ... ! 237: */ ! 238: IFDEBUG(D_OUTPUT) ! 239: printf("clnp_output: ck cache: clcp %x\n", clcp); ! 240: if (clcp != NULL) { ! 241: printf("\tclc_dst %s\n", clnp_iso_addrp(&clcp->clc_dst)); ! 242: printf("\tisop_opts x%x, clc_opts x%x\n", isop->isop_options, ! 243: clcp->clc_options); ! 244: if (isop->isop_route.ro_rt) ! 245: printf("\tro_rt x%x, rt_flags x%x\n", ! 246: isop->isop_route.ro_rt, isop->isop_route.ro_rt->rt_flags); ! 247: printf("\tflags x%x, clc_flags x%x\n", flags, clcp->clc_flags); ! 248: printf("\tclc_hdr x%x\n", clcp->clc_hdr); ! 249: } ! 250: ENDDEBUG ! 251: if ((clcp != NULL) && /* cache exists */ ! 252: (isop->isop_options == clcp->clc_options) && /* same options */ ! 253: (iso_addrmatch1(dst, &clcp->clc_dst)) && /* dst still same */ ! 254: (isop->isop_route.ro_rt != NULL) && /* route exists */ ! 255: (isop->isop_route.ro_rt == clcp->clc_rt) && /* and is cached */ ! 256: (isop->isop_route.ro_rt->rt_flags & RTF_UP) && /* route still up */ ! 257: (flags == clcp->clc_flags) && /* same flags */ ! 258: (clcp->clc_hdr != NULL)) { /* hdr mbuf exists */ ! 259: /* ! 260: * The cache is valid ! 261: */ ! 262: ! 263: IFDEBUG(D_OUTPUT) ! 264: printf("clnp_output: using cache\n"); ! 265: ENDDEBUG ! 266: ! 267: m = m_copy(clcp->clc_hdr, 0, (int)M_COPYALL); ! 268: if (m == NULL) { ! 269: /* ! 270: * No buffers left to copy cached packet header. Use ! 271: * the cached packet header this time, and ! 272: * mark the hdr as vacant ! 273: */ ! 274: m = clcp->clc_hdr; ! 275: clcp->clc_hdr = NULL; ! 276: } ! 277: m->m_next = m0; /* ASSUMES pkt hdr is 1 mbuf long */ ! 278: clnp = mtod(m, struct clnp_fixed *); ! 279: } else { ! 280: struct clnp_optidx *oidx = NULL; /* index to clnp options */ ! 281: ! 282: /* ! 283: * The cache is not valid. Allocate an mbuf (if necessary) ! 284: * to hold cached info. If one is not available, then ! 285: * don't bother with the cache ! 286: */ ! 287: INCSTAT(cns_cachemiss); ! 288: if (flags & CLNP_NOCACHE) { ! 289: clcp = &clc; ! 290: } else { ! 291: if (isop->isop_clnpcache == NULL) { ! 292: /* ! 293: * There is no clnpcache. Allocate an mbuf to hold one ! 294: */ ! 295: if ((isop->isop_clnpcache = m_get(M_DONTWAIT, MT_HEADER)) ! 296: == NULL) { ! 297: /* ! 298: * No mbufs available. Pretend that we don't want ! 299: * caching this time. ! 300: */ ! 301: IFDEBUG(D_OUTPUT) ! 302: printf("clnp_output: no mbufs to allocate to cache\n"); ! 303: ENDDEBUG ! 304: flags |= CLNP_NOCACHE; ! 305: clcp = &clc; ! 306: } else { ! 307: clcp = mtod(isop->isop_clnpcache, struct clnp_cache *); ! 308: } ! 309: } else { ! 310: /* ! 311: * A clnpcache mbuf exists. If the clc_hdr is not null, ! 312: * we must free it, as a new one is about to be created. ! 313: */ ! 314: clcp = mtod(isop->isop_clnpcache, struct clnp_cache *); ! 315: if (clcp->clc_hdr != NULL) { ! 316: /* ! 317: * The clc_hdr is not null but a clnpcache mbuf exists. ! 318: * This means that there was a cache, but the existing ! 319: * copy of the hdr is no longer valid. Free it now ! 320: * before we lose the pointer to it. ! 321: */ ! 322: IFDEBUG(D_OUTPUT) ! 323: printf("clnp_output: freeing old clc_hdr 0x%x\n", ! 324: clcp->clc_hdr); ! 325: ENDDEBUG ! 326: m_free(clcp->clc_hdr); ! 327: IFDEBUG(D_OUTPUT) ! 328: printf("clnp_output: freed old clc_hdr (done)\n"); ! 329: ENDDEBUG ! 330: } ! 331: } ! 332: } ! 333: IFDEBUG(D_OUTPUT) ! 334: printf("clnp_output: NEW clcp x%x\n",clcp); ! 335: ENDDEBUG ! 336: bzero((caddr_t)clcp, sizeof(struct clnp_cache)); ! 337: ! 338: if (isop->isop_optindex) ! 339: oidx = mtod(isop->isop_optindex, struct clnp_optidx *); ! 340: ! 341: /* ! 342: * Don't allow packets with security, quality of service, ! 343: * priority, or error report options to be sent. ! 344: */ ! 345: if ((isop->isop_options) && (oidx)) { ! 346: if ((oidx->cni_securep) || ! 347: (oidx->cni_priorp) || ! 348: (oidx->cni_qos_formatp) || ! 349: (oidx->cni_er_reason != ER_INVALREAS)) { ! 350: IFDEBUG(D_OUTPUT) ! 351: printf("clnp_output: pkt dropped - option unsupported\n"); ! 352: ENDDEBUG ! 353: m_freem(m0); ! 354: return(EINVAL); ! 355: } ! 356: } ! 357: ! 358: /* ! 359: * Don't allow any invalid flags to be set ! 360: */ ! 361: if ((flags & (CLNP_VFLAGS)) != flags) { ! 362: IFDEBUG(D_OUTPUT) ! 363: printf("clnp_output: packet dropped - flags unsupported\n"); ! 364: ENDDEBUG ! 365: INCSTAT(cns_odropped); ! 366: m_freem(m0); ! 367: return(EINVAL); ! 368: } ! 369: ! 370: /* ! 371: * Don't allow funny lengths on dst; src may be zero in which ! 372: * case we insert the source address based upon the interface ! 373: */ ! 374: if ((src->isoa_len > sizeof(struct iso_addr)) || ! 375: (dst->isoa_len == 0) || ! 376: (dst->isoa_len > sizeof(struct iso_addr))) { ! 377: m_freem(m0); ! 378: INCSTAT(cns_odropped); ! 379: return(ENAMETOOLONG); ! 380: } ! 381: ! 382: /* ! 383: * Grab mbuf to contain header ! 384: */ ! 385: MGETHDR(m, M_DONTWAIT, MT_HEADER); ! 386: if (m == 0) { ! 387: m_freem(m0); ! 388: INCSTAT(cns_odropped); ! 389: return(ENOBUFS); ! 390: } ! 391: INCSTAT(cns_sent); ! 392: m->m_next = m0; ! 393: clnp = mtod(m, struct clnp_fixed *); ! 394: clcp->clc_segoff = 0; ! 395: ! 396: /* ! 397: * Fill in all of fixed hdr except lengths and checksum ! 398: */ ! 399: if (flags & CLNP_SEND_RAW) { ! 400: *clnp = raw_template; ! 401: } else if (flags & CLNP_ECHO) { ! 402: *clnp = echo_template; ! 403: } else if (flags & CLNP_ECHOR) { ! 404: *clnp = echor_template; ! 405: } else { ! 406: *clnp = dt_template; ! 407: } ! 408: if (flags & CLNP_NO_SEG) ! 409: clnp->cnf_type &= ~CNF_SEG_OK; ! 410: if (flags & CLNP_NO_ER) ! 411: clnp->cnf_type &= ~CNF_ERR_OK; ! 412: ! 413: /* ! 414: * Route packet; special case for source rt ! 415: */ ! 416: if ((isop->isop_options) && CLNPSRCRT_VALID(oidx)) { ! 417: IFDEBUG(D_OUTPUT) ! 418: printf("clnp_output: calling clnp_srcroute\n"); ! 419: ENDDEBUG ! 420: error = clnp_srcroute(isop->isop_options, oidx, &isop->isop_route, ! 421: &clcp->clc_firsthop, &clcp->clc_ifa, dst); ! 422: } else { ! 423: IFDEBUG(D_OUTPUT) ! 424: ENDDEBUG ! 425: error = clnp_route(dst, &isop->isop_route, flags, ! 426: &clcp->clc_firsthop, &clcp->clc_ifa); ! 427: } ! 428: if (error || (clcp->clc_ifa == 0)) { ! 429: IFDEBUG(D_OUTPUT) ! 430: printf("clnp_output: route failed, errno %d\n", error); ! 431: printf("@clcp:\n"); ! 432: dump_buf(clcp, sizeof (struct clnp_cache)); ! 433: ENDDEBUG ! 434: goto bad; ! 435: } ! 436: clcp->clc_rt = isop->isop_route.ro_rt; /* XXX */ ! 437: clcp->clc_ifp = clcp->clc_ifa->ia_ifp; /* XXX */ ! 438: ! 439: IFDEBUG(D_OUTPUT) ! 440: printf("clnp_output: packet routed to %s\n", ! 441: clnp_iso_addrp( ! 442: &((struct sockaddr_iso *)clcp->clc_firsthop)->siso_addr)); ! 443: ENDDEBUG ! 444: ! 445: /* ! 446: * If src address is not yet specified, use address of ! 447: * interface. NOTE: this will now update the laddr field in ! 448: * the isopcb. Is this desirable? RAH? ! 449: */ ! 450: if (src->isoa_len == 0) { ! 451: src = &(clcp->clc_ifa->ia_addr.siso_addr); ! 452: IFDEBUG(D_OUTPUT) ! 453: printf("clnp_output: new src %s\n", clnp_iso_addrp(src)); ! 454: ENDDEBUG ! 455: } ! 456: ! 457: /* ! 458: * Insert the source and destination address, ! 459: */ ! 460: hoff = (caddr_t)clnp + sizeof(struct clnp_fixed); ! 461: CLNP_INSERT_ADDR(hoff, *dst); ! 462: CLNP_INSERT_ADDR(hoff, *src); ! 463: ! 464: /* ! 465: * Leave room for the segment part, if segmenting is selected ! 466: */ ! 467: if (clnp->cnf_type & CNF_SEG_OK) { ! 468: clcp->clc_segoff = hoff - (caddr_t)clnp; ! 469: hoff += sizeof(struct clnp_segment); ! 470: } ! 471: ! 472: clnp->cnf_hdr_len = m->m_len = (u_char)(hoff - (caddr_t)clnp); ! 473: hdrlen = clnp->cnf_hdr_len; ! 474: ! 475: #ifdef DECBIT ! 476: /* ! 477: * Add the globally unique QOS (with room for congestion experienced ! 478: * bit). I can safely assume that this option is not in the options ! 479: * mbuf below because I checked that the option was not specified ! 480: * previously ! 481: */ ! 482: if ((m->m_len + sizeof(qos_option)) < MLEN) { ! 483: bcopy((caddr_t)qos_option, hoff, sizeof(qos_option)); ! 484: clnp->cnf_hdr_len += sizeof(qos_option); ! 485: hdrlen += sizeof(qos_option); ! 486: m->m_len += sizeof(qos_option); ! 487: } ! 488: #endif /* DECBIT */ ! 489: ! 490: /* ! 491: * If an options mbuf is present, concatenate a copy to the hdr mbuf. ! 492: */ ! 493: if (isop->isop_options) { ! 494: struct mbuf *opt_copy = m_copy(isop->isop_options, 0, (int)M_COPYALL); ! 495: if (opt_copy == NULL) { ! 496: error = ENOBUFS; ! 497: goto bad; ! 498: } ! 499: /* Link in place */ ! 500: opt_copy->m_next = m->m_next; ! 501: m->m_next = opt_copy; ! 502: ! 503: /* update size of header */ ! 504: clnp->cnf_hdr_len += opt_copy->m_len; ! 505: hdrlen += opt_copy->m_len; ! 506: } ! 507: ! 508: if (hdrlen > CLNP_HDR_MAX) { ! 509: error = EMSGSIZE; ! 510: goto bad; ! 511: } ! 512: ! 513: /* ! 514: * Now set up the cache entry in the pcb ! 515: */ ! 516: if ((flags & CLNP_NOCACHE) == 0) { ! 517: if (clcp->clc_hdr = m_copy(m, 0, (int)clnp->cnf_hdr_len)) { ! 518: clcp->clc_dst = *dst; ! 519: clcp->clc_flags = flags; ! 520: clcp->clc_options = isop->isop_options; ! 521: } ! 522: } ! 523: } ! 524: /* ! 525: * If small enough for interface, send directly ! 526: * Fill in segmentation part of hdr if using the full protocol ! 527: */ ! 528: total_len = clnp->cnf_hdr_len + datalen; ! 529: if (clnp->cnf_type & CNF_SEG_OK) { ! 530: struct clnp_segment seg_part; /* segment part of hdr */ ! 531: seg_part.cng_id = htons(clnp_id++); ! 532: seg_part.cng_off = htons(0); ! 533: seg_part.cng_tot_len = htons(total_len); ! 534: (void) bcopy((caddr_t)&seg_part, (caddr_t) clnp + clcp->clc_segoff, ! 535: sizeof(seg_part)); ! 536: } ! 537: if (total_len <= SN_MTU(clcp->clc_ifp, clcp->clc_rt)) { ! 538: HTOC(clnp->cnf_seglen_msb, clnp->cnf_seglen_lsb, total_len); ! 539: m->m_pkthdr.len = total_len; ! 540: /* ! 541: * Compute clnp checksum (on header only) ! 542: */ ! 543: if (flags & CLNP_NO_CKSUM) { ! 544: HTOC(clnp->cnf_cksum_msb, clnp->cnf_cksum_lsb, 0); ! 545: } else { ! 546: iso_gen_csum(m, CLNP_CKSUM_OFF, (int)clnp->cnf_hdr_len); ! 547: } ! 548: ! 549: IFDEBUG(D_DUMPOUT) ! 550: struct mbuf *mdump = m; ! 551: printf("clnp_output: sending dg:\n"); ! 552: while (mdump != NULL) { ! 553: dump_buf(mtod(mdump, caddr_t), mdump->m_len); ! 554: mdump = mdump->m_next; ! 555: } ! 556: ENDDEBUG ! 557: ! 558: error = SN_OUTPUT(clcp, m); ! 559: goto done; ! 560: } else { ! 561: /* ! 562: * Too large for interface; fragment if possible. ! 563: */ ! 564: error = clnp_fragment(clcp->clc_ifp, m, clcp->clc_firsthop, ! 565: total_len, clcp->clc_segoff, flags, clcp->clc_rt); ! 566: goto done; ! 567: } ! 568: bad: ! 569: m_freem(m); ! 570: done: ! 571: if (error) { ! 572: clnp_stat.cns_sent--; ! 573: clnp_stat.cns_odropped++; ! 574: } ! 575: return (error); ! 576: } ! 577: ! 578: int clnp_ctloutput() ! 579: { ! 580: }
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.