|
|
1.1 root 1: /***********************************************************
2: Copyright IBM Corporation 1987
3:
4: All Rights Reserved
5:
6: Permission to use, copy, modify, and distribute this software and its
7: documentation for any purpose and without fee is hereby granted,
8: provided that the above copyright notice appear in all copies and that
9: both that copyright notice and this permission notice appear in
10: supporting documentation, and that the name of IBM not be
11: used in advertising or publicity pertaining to distribution of the
12: software without specific, written prior permission.
13:
14: IBM DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
15: ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
16: IBM BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
17: ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
18: WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
19: ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
20: SOFTWARE.
21:
22: ******************************************************************/
23:
24: /*
25: * ARGO Project, Computer Sciences Dept., University of Wisconsin - Madison
26: */
27: /*
28: * ARGO TP
29: *
30: * $Header: tp_output.c,v 5.4 88/11/18 17:28:08 nhall Exp $
31: * $Source: /usr/argo/sys/netiso/RCS/tp_output.c,v $
32: * @(#)tp_output.c 7.7 (Berkeley) 6/29/90 *
33: *
34: * In here is tp_ctloutput(), the guy called by [sg]etsockopt(),
35: */
36:
37: #ifndef lint
38: static char *rcsid = "$Header: tp_output.c,v 5.4 88/11/18 17:28:08 nhall Exp $";
39: #endif lint
40:
41: #include "param.h"
42: #include "mbuf.h"
43: #include "systm.h"
44: #include "socket.h"
45: #include "socketvar.h"
46: #include "protosw.h"
47: #include "user.h"
48: #include "kernel.h"
49: #include "errno.h"
50: #include "time.h"
51: #include "tp_param.h"
52: #include "tp_user.h"
53: #include "tp_stat.h"
54: #include "tp_ip.h"
55: #include "tp_clnp.h"
56: #include "tp_timer.h"
57: #include "argo_debug.h"
58: #include "tp_pcb.h"
59: #include "tp_trace.h"
60:
61: #define USERFLAGSMASK_G 0x0f00643b
62: #define USERFLAGSMASK_S 0x0f000432
63: #define TPDUSIZESHIFT 24
64: #define CLASSHIFT 16
65:
66: /*
67: * NAME: tp_consistency()
68: *
69: * CALLED FROM:
70: * tp_ctloutput(), tp_input()
71: *
72: * FUNCTION and ARGUMENTS:
73: * Checks the consistency of options and tpdusize with class,
74: * using the parameters passed in via (param).
75: * (cmd) may be TP_STRICT or TP_FORCE or both.
76: * Force means it will set all the values in (tpcb) to those in
77: * the input arguements iff no errors were encountered.
78: * Strict means that no inconsistency will be tolerated. If it's
79: * not used, checksum and tpdusize inconsistencies will be tolerated.
80: * The reason for this is that in some cases, when we're negotiating down
81: * from class 4, these options should be changed but should not
82: * cause negotiation to fail.
83: *
84: * RETURNS
85: * E* or EOK
86: * E* if the various parms aren't ok for a given class
87: * EOK if they are ok for a given class
88: */
89:
90: int
91: tp_consistency( tpcb, cmd, param )
92: u_int cmd;
93: struct tp_conn_param *param;
94: struct tp_pcb *tpcb;
95: {
96: register int error = EOK;
97: int class_to_use = tp_mask_to_num(param->p_class);
98:
99: IFTRACE(D_SETPARAMS)
100: tptrace(TPPTmisc,
101: "tp_consist enter class_to_use dontchange param.class cmd",
102: class_to_use, param->p_dont_change_params, param->p_class, cmd);
103: ENDTRACE
104: IFDEBUG(D_SETPARAMS)
105: printf("tp_consistency %s %s\n",
106: cmd& TP_FORCE? "TP_FORCE": "",
107: cmd& TP_STRICT? "TP_STRICT":"");
108: ENDDEBUG
109: if ((cmd & TP_FORCE) && (param->p_dont_change_params)) {
110: cmd &= ~TP_FORCE;
111: }
112: /* can switch net services within a domain, but
113: * cannot switch domains
114: */
115: switch( param->p_netservice) {
116: case ISO_CONS:
117: case ISO_CLNS:
118: case ISO_COSNS:
119: /* param->p_netservice in ISO DOMAIN */
120: if(tpcb->tp_domain != AF_ISO ) {
121: error = EINVAL; goto done;
122: }
123: break;
124: case IN_CLNS:
125: /* param->p_netservice in INET DOMAIN */
126: if( tpcb->tp_domain != AF_INET ) {
127: error = EINVAL; goto done;
128: }
129: break;
130: /* no others not possible-> netservice is a 2-bit field! */
131: }
132:
133: IFDEBUG(D_SETPARAMS)
134: printf("p_class 0x%x, class_to_use 0x%x\n", param->p_class,
135: class_to_use);
136: ENDDEBUG
137: if((param->p_netservice < 0) || (param->p_netservice > TP_MAX_NETSERVICES)){
138: error = EINVAL; goto done;
139: }
140: if( (param->p_class & TP_CLASSES_IMPLEMENTED) == 0 ) {
141: error = EINVAL; goto done;
142: }
143: IFDEBUG(D_SETPARAMS)
144: printf("Nretrans 0x%x\n", param->p_Nretrans );
145: ENDDEBUG
146: if( ( param->p_Nretrans < 1 ) ||
147: (param->p_cr_ticks < 1) || (param->p_cc_ticks < 1) ) {
148: /* bad for any class because negot has to be done a la class 4 */
149: error = EINVAL; goto done;
150: }
151: IFDEBUG(D_SETPARAMS)
152: printf("winsize 0x%x\n", param->p_winsize );
153: ENDDEBUG
154: if( (param->p_winsize < 128 ) ||
155: (param->p_winsize < param->p_tpdusize ) ||
156: (param->p_winsize > ((1+SB_MAX)>>2 /* 1/4 of the max */)) ) {
157: error = EINVAL; goto done;
158: } else {
159: if( tpcb->tp_state == TP_CLOSED )
160: soreserve(tpcb->tp_sock, (u_long)param->p_winsize,
161: (u_long)param->p_winsize);
162: }
163: IFDEBUG(D_SETPARAMS)
164: printf("use_csum 0x%x\n", param->p_use_checksum );
165: printf("xtd_format 0x%x\n", param->p_xtd_format );
166: printf("xpd_service 0x%x\n", param->p_xpd_service );
167: printf("tpdusize 0x%x\n", param->p_tpdusize );
168: printf("tpcb->flags 0x%x\n", tpcb->tp_flags );
169: ENDDEBUG
170: switch( class_to_use ) {
171:
172: case 0:
173: /* do not use checksums, xtd format, or XPD */
174:
175: if( param->p_use_checksum | param->p_xtd_format | param->p_xpd_service ) {
176: if(cmd & TP_STRICT) {
177: error = EINVAL;
178: } else {
179: param->p_use_checksum = 0;
180: param->p_xtd_format = 0;
181: param->p_xpd_service = 0;
182: }
183: break;
184: }
185:
186: if (param->p_tpdusize < TP_MIN_TPDUSIZE) {
187: if(cmd & TP_STRICT) {
188: error = EINVAL;
189: } else {
190: param->p_tpdusize = TP_MIN_TPDUSIZE;
191: }
192: break;
193: }
194: if (param->p_tpdusize > TP0_TPDUSIZE) {
195: if (cmd & TP_STRICT) {
196: error = EINVAL;
197: } else {
198: param->p_tpdusize = TP0_TPDUSIZE;
199: }
200: break;
201: }
202:
203: /* connect/disc data not allowed for class 0 */
204: if (tpcb->tp_ucddata) {
205: if(cmd & TP_STRICT) {
206: error = EINVAL;
207: } else if(cmd & TP_FORCE) {
208: m_freem(tpcb->tp_ucddata);
209: tpcb->tp_ucddata = 0;
210: }
211: }
212: break;
213:
214: case 4:
215: IFDEBUG(D_SETPARAMS)
216: printf("dt_ticks 0x%x\n", param->p_dt_ticks );
217: printf("x_ticks 0x%x\n", param->p_x_ticks );
218: printf("dr_ticks 0x%x\n", param->p_dr_ticks );
219: printf("keepalive 0x%x\n", param->p_keepalive_ticks );
220: printf("sendack 0x%x\n", param->p_sendack_ticks );
221: printf("inact 0x%x\n", param->p_inact_ticks );
222: printf("ref 0x%x\n", param->p_ref_ticks );
223: ENDDEBUG
224: if( (param->p_class & TP_CLASS_4 ) && (
225: (param->p_dt_ticks < 1) || (param->p_dr_ticks < 1) ||
226: (param->p_x_ticks < 1) || (param->p_keepalive_ticks < 1) ||
227: (param->p_sendack_ticks < 1) || (param->p_ref_ticks < 1) ||
228: (param->p_inact_ticks < 1) ) ) {
229: error = EINVAL;
230: break;
231: }
232: IFDEBUG(D_SETPARAMS)
233: printf("rx_strat 0x%x\n", param->p_rx_strat );
234: ENDDEBUG
235: if(param->p_rx_strat >
236: ( TPRX_USE_CW | TPRX_EACH | TPRX_FASTSTART) ) {
237: if(cmd & TP_STRICT) {
238: error = EINVAL;
239: } else {
240: param->p_rx_strat = TPRX_USE_CW;
241: }
242: break;
243: }
244: IFDEBUG(D_SETPARAMS)
245: printf("ack_strat 0x%x\n", param->p_ack_strat );
246: ENDDEBUG
247: if((param->p_ack_strat != 0) && (param->p_ack_strat != 1)) {
248: if(cmd & TP_STRICT) {
249: error = EINVAL;
250: } else {
251: param->p_ack_strat = TPACK_WINDOW;
252: }
253: break;
254: }
255: if (param->p_tpdusize < TP_MIN_TPDUSIZE) {
256: if(cmd & TP_STRICT) {
257: error = EINVAL;
258: } else {
259: param->p_tpdusize = TP_MIN_TPDUSIZE;
260: }
261: break;
262: }
263: if (param->p_tpdusize > TP_TPDUSIZE) {
264: if(cmd & TP_STRICT) {
265: error = EINVAL;
266: } else {
267: param->p_tpdusize = TP_TPDUSIZE;
268: }
269: break;
270: }
271: break;
272: }
273:
274: if ((error==0) && (cmd & TP_FORCE)) {
275: tpcb->tp_tpdusize = param->p_tpdusize;
276: tpcb->tp_class = param->p_class;
277: tpcb->tp_use_checksum = param->p_use_checksum;
278: tpcb->tp_xpd_service = param->p_xpd_service;
279: tpcb->tp_xtd_format = param->p_xtd_format;
280: }
281:
282: done:
283:
284: IFTRACE(D_CONN)
285: tptrace(TPPTmisc, "tp_consist returns class xtdfmt cmd",
286: error, tpcb->tp_class, tpcb->tp_xtd_format, cmd);
287: ENDTRACE
288: IFDEBUG(D_CONN)
289: printf(
290: "tp_consist rtns 0x%x class 0x%x xtd_fmt 0x%x cmd 0x%x\n",
291: error, tpcb->tp_class, tpcb->tp_xtd_format, cmd);
292: ENDDEBUG
293: return error;
294: }
295:
296: /*
297: * NAME: tp_ctloutput()
298: *
299: * CALLED FROM:
300: * [sg]etsockopt(), via so[sg]etopt().
301: *
302: * FUNCTION and ARGUMENTS:
303: * Implements the socket options at transport level.
304: * (cmd) is either PRCO_SETOPT or PRCO_GETOPT (see ../sys/protosw.h).
305: * (so) is the socket.
306: * (level) is SOL_TRANSPORT (see ../sys/socket.h)
307: * (optname) is the particular command or option to be set.
308: * (**mp) is an mbuf structure.
309: *
310: * RETURN VALUE:
311: * ENOTSOCK if the socket hasn't got an associated tpcb
312: * EINVAL if
313: * trying to set window too big
314: * trying to set illegal max tpdu size
315: * trying to set illegal credit fraction
316: * trying to use unknown or unimplemented class of TP
317: * structure passed to set timer values is wrong size
318: * illegal combination of command/GET-SET option,
319: * e.g., GET w/ TPOPT_CDDATA_CLEAR:
320: * EOPNOTSUPP if the level isn't transport, or command is neither GET nor SET
321: * or if the transport-specific command is not implemented
322: * EISCONN if trying a command that isn't allowed after a connection
323: * is established
324: * ENOTCONN if trying a command that is allowed only if a connection is
325: * established
326: * EMSGSIZE if trying to give too much data on connect/disconnect
327: *
328: * SIDE EFFECTS:
329: *
330: * NOTES:
331: */
332: ProtoHook
333: tp_ctloutput(cmd, so, level, optname, mp)
334: int cmd, level, optname;
335: struct socket *so;
336: struct mbuf **mp;
337: {
338: struct tp_pcb *tpcb = sototpcb(so);
339: int s = splnet();
340: caddr_t value;
341: unsigned val_len;
342: int error = 0;
343:
344: IFTRACE(D_REQUEST)
345: tptrace(TPPTmisc, "tp_ctloutput cmd so optname mp",
346: cmd, so, optname, mp);
347: ENDTRACE
348: IFDEBUG(D_REQUEST)
349: printf(
350: "tp_ctloutput so 0x%x cmd 0x%x optname 0x%x, mp 0x%x *mp 0x%x tpcb 0x%x\n",
351: so, cmd, optname, mp, mp?*mp:0, tpcb);
352: ENDDEBUG
353: if( tpcb == (struct tp_pcb *)0 ) {
354: error = ENOTSOCK; goto done;
355: }
356: if(*mp == MNULL) {
357: register struct mbuf *m;
358:
359: MGET(m, M_DONTWAIT, TPMT_SONAME); /* does off, type, next */
360: if (m == NULL) {
361: splx(s);
362: return ENOBUFS;
363: }
364: m->m_len = 0;
365: m->m_act = 0;
366: *mp = m;
367: }
368:
369: /*
370: * Hook so one can set network options via a tp socket.
371: */
372: if ( level == SOL_NETWORK ) {
373: if ((tpcb->tp_nlproto == NULL) || (tpcb->tp_npcb == NULL))
374: error = ENOTSOCK;
375: else if (tpcb->tp_nlproto->nlp_ctloutput == NULL)
376: error = EOPNOTSUPP;
377: else
378: error = (tpcb->tp_nlproto->nlp_ctloutput)(cmd, optname,
379: tpcb->tp_npcb, *mp);
380: goto done;
381: } else if ( level != SOL_TRANSPORT ) {
382: error = EOPNOTSUPP; goto done;
383: }
384: if (cmd != PRCO_GETOPT && cmd != PRCO_SETOPT) {
385: error = EOPNOTSUPP; goto done;
386: }
387: if ( so->so_error ) {
388: error = so->so_error; goto done;
389: }
390:
391: /* The only options allowed after connection is established
392: * are GET (anything) and SET DISC DATA and SET PERF MEAS
393: */
394: if ( ((so->so_state & SS_ISCONNECTING)||(so->so_state & SS_ISCONNECTED))
395: &&
396: (cmd == PRCO_SETOPT &&
397: optname != TPOPT_DISC_DATA &&
398: optname != TPOPT_CFRM_DATA &&
399: optname != TPOPT_PERF_MEAS &&
400: optname != TPOPT_CDDATA_CLEAR ) ) {
401: error = EISCONN; goto done;
402: }
403: /* The only options allowed after disconnection are GET DISC DATA,
404: * and TPOPT_PSTATISTICS
405: * and they're not allowed if the ref timer has gone off, because
406: * the tpcb is gone
407: */
408: if ((so->so_state & (SS_ISCONNECTED | SS_ISCONFIRMING)) == 0) {
409: if ( so->so_tpcb == (caddr_t)0 ) {
410: error = ENOTCONN; goto done;
411: }
412: if ( (tpcb->tp_state == TP_REFWAIT || tpcb->tp_state == TP_CLOSING) &&
413: (optname != TPOPT_DISC_DATA && optname != TPOPT_PSTATISTICS)) {
414: error = ENOTCONN; goto done;
415: }
416: }
417:
418: value = mtod(*mp, caddr_t); /* it's aligned, don't worry,
419: * but lint complains about it
420: */
421: val_len = (*mp)->m_len;
422:
423: switch (optname) {
424:
425: case TPOPT_INTERCEPT:
426: if (error = suser(u.u_cred, &u.u_acflag))
427: break;
428: else if (cmd != PRCO_SETOPT || tpcb->tp_state != TP_LISTENING)
429: error = EINVAL;
430: else {
431: register struct tp_pcb *t = 0;
432: struct mbuf *m = m_getclr(M_WAIT, MT_SONAME);
433: struct sockaddr *sa = mtod(m, struct sockaddr *);
434: (*tpcb->tp_nlproto->nlp_getnetaddr)(tpcb->tp_npcb, m, TP_LOCAL);
435: switch (sa->sa_family) {
436: case AF_ISO:
437: if (((struct sockaddr_iso *)sa)->siso_nlen == 0)
438: default: error = EINVAL;
439: break;
440: case AF_INET:
441: if (((struct sockaddr_in *)sa)->sin_addr.s_addr == 0)
442: error = EINVAL;
443: break;
444: }
445: for (t = tp_intercepts; t; t = t->tp_nextlisten) {
446: if (t->tp_nlproto->nlp_afamily != tpcb->tp_nlproto->nlp_afamily)
447: continue;
448: if ((*t->tp_nlproto->nlp_cmpnetaddr)(t->tp_npcb, sa, TP_LOCAL))
449: error = EADDRINUSE;
450: }
451: m_freem(m);
452: if (error)
453: break;
454: }
455: {
456: register struct tp_pcb **tt;
457: for (tt = &tp_listeners; *tt; tt = &((*tt)->tp_nextlisten))
458: if (*tt == tpcb)
459: break;
460: if (*tt)
461: *tt = tpcb->tp_nextlisten;
462: else
463: {error = EHOSTUNREACH; goto done; }
464: }
465: tpcb->tp_nextlisten = tp_intercepts;
466: tp_intercepts = tpcb;
467: break;
468:
469: case TPOPT_MY_TSEL:
470: if ( cmd == PRCO_GETOPT ) {
471: ASSERT( tpcb->tp_lsuffixlen <= MAX_TSAP_SEL_LEN );
472: bcopy((caddr_t)tpcb->tp_lsuffix, value, tpcb->tp_lsuffixlen);
473: (*mp)->m_len = tpcb->tp_lsuffixlen;
474: } else /* cmd == PRCO_SETOPT */ {
475: if( (val_len > MAX_TSAP_SEL_LEN) || (val_len <= 0 )) {
476: printf("val_len 0x%x (*mp)->m_len 0x%x\n", val_len, (*mp));
477: error = EINVAL;
478: } else {
479: bcopy(value, (caddr_t)tpcb->tp_lsuffix, val_len);
480: tpcb->tp_lsuffixlen = val_len;
481: }
482: }
483: break;
484:
485: case TPOPT_PEER_TSEL:
486: if ( cmd == PRCO_GETOPT ) {
487: ASSERT( tpcb->tp_fsuffixlen <= MAX_TSAP_SEL_LEN );
488: bcopy((caddr_t)tpcb->tp_fsuffix, value, tpcb->tp_fsuffixlen);
489: (*mp)->m_len = tpcb->tp_fsuffixlen;
490: } else /* cmd == PRCO_SETOPT */ {
491: if( (val_len > MAX_TSAP_SEL_LEN) || (val_len <= 0 )) {
492: printf("val_len 0x%x (*mp)->m_len 0x%x\n", val_len, (*mp));
493: error = EINVAL;
494: } else {
495: bcopy(value, (caddr_t)tpcb->tp_fsuffix, val_len);
496: tpcb->tp_fsuffixlen = val_len;
497: }
498: }
499: break;
500:
501: case TPOPT_FLAGS:
502: IFDEBUG(D_REQUEST)
503: printf("%s TPOPT_FLAGS value 0x%x *value 0x%x, flags 0x%x \n",
504: cmd==PRCO_GETOPT?"GET":"SET",
505: value,
506: *value,
507: tpcb->tp_flags);
508: ENDDEBUG
509:
510: if ( cmd == PRCO_GETOPT ) {
511: *(int *)value = (int)tpcb->tp_flags;
512: (*mp)->m_len = sizeof(u_int);
513: } else /* cmd == PRCO_SETOPT */ {
514: error = EINVAL; goto done;
515: }
516: break;
517:
518: case TPOPT_PARAMS:
519: /* This handles:
520: * timer values,
521: * class, use of transport expedited data,
522: * max tpdu size, checksum, xtd format and
523: * disconnect indications, and may get rid of connect/disc data
524: */
525: IFDEBUG(D_SETPARAMS)
526: printf("TPOPT_PARAMS value 0x%x, cmd %s \n", value,
527: cmd==PRCO_GETOPT?"GET":"SET");
528: ENDDEBUG
529: IFDEBUG(D_REQUEST)
530: printf("TPOPT_PARAMS value 0x%x, cmd %s \n", value,
531: cmd==PRCO_GETOPT?"GET":"SET");
532: ENDDEBUG
533:
534: if ( cmd == PRCO_GETOPT ) {
535: *(struct tp_conn_param *)value = tpcb->_tp_param;
536: (*mp)->m_len = sizeof(tpcb->_tp_param);
537: } else /* cmd == PRCO_SETOPT */ {
538: if( (error =
539: tp_consistency(tpcb, TP_STRICT | TP_FORCE,
540: (struct tp_conn_param *)value))==0) {
541: /*
542: * tp_consistency doesn't copy the whole set of params
543: */
544: tpcb->_tp_param = *(struct tp_conn_param *)value;
545: (*mp)->m_len = sizeof(tpcb->_tp_param);
546: }
547: }
548: break;
549:
550: case TPOPT_PSTATISTICS:
551: #ifdef TP_PERF_MEAS
552: if (cmd == PRCO_SETOPT) {
553: error = EINVAL; goto done;
554: }
555: IFPERF(tpcb)
556: if (*mp) {
557: struct mbuf * n;
558: do {
559: MFREE(*mp, n);
560: *mp = n;
561: } while (n);
562: }
563: *mp = m_copym(tpcb->tp_p_mbuf, (int)M_COPYALL, M_WAITOK);
564: ENDPERF
565: else {
566: error = EINVAL; goto done;
567: }
568: break;
569: #else
570: error = EOPNOTSUPP;
571: goto done;
572: #endif TP_PERF_MEAS
573:
574: case TPOPT_CDDATA_CLEAR:
575: if (cmd == PRCO_GETOPT) {
576: error = EINVAL;
577: } else {
578: if (tpcb->tp_ucddata) {
579: m_freem(tpcb->tp_ucddata);
580: tpcb->tp_ucddata = 0;
581: }
582: }
583: break;
584:
585: case TPOPT_CFRM_DATA:
586: case TPOPT_DISC_DATA:
587: case TPOPT_CONN_DATA:
588: if( tpcb->tp_class == TP_CLASS_0 ) {
589: error = EOPNOTSUPP;
590: break;
591: }
592: IFDEBUG(D_REQUEST)
593: printf("%s\n", optname==TPOPT_DISC_DATA?"DISC data":"CONN data");
594: printf("m_len 0x%x, vallen 0x%x so_snd.cc 0x%x\n",
595: (*mp)->m_len, val_len, so->so_snd.sb_cc);
596: dump_mbuf(so->so_snd.sb_mb, "tp_ctloutput: sosnd ");
597: ENDDEBUG
598: if (cmd == PRCO_SETOPT) {
599: int len = tpcb->tp_ucddata ? tpcb->tp_ucddata->m_len : 0;
600: /* can append connect data in several calls */
601: if (len + val_len >
602: (optname==TPOPT_CONN_DATA?TP_MAX_CR_DATA:TP_MAX_DR_DATA) ) {
603: error = EMSGSIZE; goto done;
604: }
605: (*mp)->m_next = MNULL;
606: (*mp)->m_act = 0;
607: if (tpcb->tp_ucddata)
608: m_cat(tpcb->tp_ucddata, *mp);
609: else
610: tpcb->tp_ucddata = *mp;
611: IFDEBUG(D_REQUEST)
612: dump_mbuf(tpcb->tp_ucddata, "tp_ctloutput after CONN_DATA");
613: ENDDEBUG
614: IFTRACE(D_REQUEST)
615: tptrace(TPPTmisc,"C/D DATA: flags snd.sbcc val_len",
616: tpcb->tp_flags, so->so_snd.sb_cc,val_len,0);
617: ENDTRACE
618: *mp = MNULL; /* prevent sosetopt from freeing it! */
619: if (optname == TPOPT_CFRM_DATA && (so->so_state & SS_ISCONFIRMING))
620: (void) tp_confirm(tpcb);
621: }
622: break;
623:
624: case TPOPT_PERF_MEAS:
625: #ifdef TP_PERF_MEAS
626: if (cmd == PRCO_GETOPT) {
627: *value = (u_int)tpcb->tp_perf_on;
628: (*mp)->m_len = sizeof(u_int);
629: } else if (cmd == PRCO_SETOPT) {
630: (*mp)->m_len = 0;
631: if ((*value) != 0 && (*value) != 1 )
632: error = EINVAL;
633: else tpcb->tp_perf_on = (*value);
634: }
635: if( tpcb->tp_perf_on )
636: error = tp_setup_perf(tpcb);
637: #else TP_PERF_MEAS
638: error = EOPNOTSUPP;
639: #endif TP_PERF_MEAS
640: break;
641:
642: default:
643: error = EOPNOTSUPP;
644: }
645:
646: done:
647: IFDEBUG(D_REQUEST)
648: dump_mbuf(so->so_snd.sb_mb, "tp_ctloutput sosnd at end");
649: dump_mbuf(*mp, "tp_ctloutput *mp");
650: ENDDEBUG
651: /*
652: * sigh: getsockopt looks only at m_len : all output data must
653: * reside in the first mbuf
654: */
655: if ( error && (*mp) != MNULL )
656: (*mp)->m_len = 0;
657: if( (*mp) != MNULL ) {
658: ASSERT ( m_compress(*mp, mp) <= MLEN );
659: IFDEBUG(D_REQUEST)
660: dump_mbuf(*mp, "tp_ctloutput *mp after compress");
661: ENDDEBUG
662: }
663:
664: splx(s);
665: return error;
666: }
This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.