|
|
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.