Merge remote-tracking branch 'origin/master'
[unleashed/lotheac.git] / usr / src / uts / common / io / rlmod.c
blob6a79f5e29511412235ee3bfe3063873c6d844a1a
1 /*
2 * CDDL HEADER START
4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License, Version 1.0 only
6 * (the "License"). You may not use this file except in compliance
7 * with the License.
9 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10 * or http://www.opensolaris.org/os/licensing.
11 * See the License for the specific language governing permissions
12 * and limitations under the License.
14 * When distributing Covered Code, include this CDDL HEADER in each
15 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16 * If applicable, add the following below this CDDL HEADER, with the
17 * fields enclosed by brackets "[]" replaced with your own identifying
18 * information: Portions Copyright [yyyy] [name of copyright owner]
20 * CDDL HEADER END
23 * Copyright 2004 Sun Microsystems, Inc. All rights reserved.
24 * Use is subject to license terms.
25 * Copyright (c) 2016 by Delphix. All rights reserved.
29 * This module implements the services provided by the rlogin daemon
30 * after the connection is set up. Mainly this means responding to
31 * interrupts and window size changes. It begins operation in "disabled"
32 * state, and sends a T_DATA_REQ to the daemon to indicate that it is
33 * in place and ready to be enabled. The daemon can then know when all
34 * data which sneaked passed rlmod (before it was pushed) has been received.
35 * The daemon may process this data, or send data back to be inserted in
36 * the read queue at the head with the RL_IOC_ENABLE ioctl.
39 #include <sys/types.h>
40 #include <sys/param.h>
41 #include <sys/stream.h>
42 #include <sys/stropts.h>
43 #include <sys/strsun.h>
44 #include <sys/kmem.h>
45 #include <sys/errno.h>
46 #include <sys/ddi.h>
47 #include <sys/sunddi.h>
48 #include <sys/tihdr.h>
49 #include <sys/ptem.h>
50 #include <sys/conf.h>
51 #include <sys/debug.h>
52 #include <sys/modctl.h>
53 #include <sys/vtrace.h>
54 #include <sys/rlioctl.h>
55 #include <sys/termios.h>
56 #include <sys/termio.h>
57 #include <sys/byteorder.h>
58 #include <sys/cmn_err.h>
59 #include <sys/cryptmod.h>
61 extern struct streamtab rloginmodinfo;
63 static struct fmodsw fsw = {
64 "rlmod",
65 &rloginmodinfo,
66 D_MTQPAIR | D_MP
70 * Module linkage information for the kernel.
73 static struct modlstrmod modlstrmod = {
74 &mod_strmodops,
75 "rloginmod module",
76 &fsw
79 static struct modlinkage modlinkage = {
80 MODREV_1, &modlstrmod, NULL
84 int
85 _init(void)
87 return (mod_install(&modlinkage));
90 int
91 _fini(void)
93 return (mod_remove(&modlinkage));
96 int
97 _info(struct modinfo *modinfop)
99 return (mod_info(&modlinkage, modinfop));
102 struct rlmod_info; /* forward reference for function prototype */
104 static int rlmodopen(queue_t *, dev_t *, int, int, cred_t *);
105 static int rlmodclose(queue_t *, int, cred_t *);
106 static int rlmodrput(queue_t *, mblk_t *);
107 static int rlmodrsrv(queue_t *);
108 static int rlmodwput(queue_t *, mblk_t *);
109 static int rlmodwsrv(queue_t *);
110 static int rlmodrmsg(queue_t *, mblk_t *);
111 static mblk_t *make_expmblk(char);
112 static int rlwinctl(queue_t *, mblk_t *);
113 static mblk_t *rlwinsetup(queue_t *, mblk_t *, unsigned char *);
115 static void rlmod_timer(void *);
116 static void rlmod_buffer(void *);
117 static boolean_t tty_flow(queue_t *, struct rlmod_info *, mblk_t *);
118 static boolean_t rlmodwioctl(queue_t *, mblk_t *);
119 static void recover(queue_t *, mblk_t *, size_t);
120 static void recover1(queue_t *, size_t);
122 #define RLMOD_ID 106
123 #define SIMWAIT (1*hz)
126 * Stream module data structure definitions.
127 * generally pushed onto tcp by rlogin daemon
130 static struct module_info rloginmodiinfo = {
131 RLMOD_ID, /* module id number */
132 "rlmod", /* module name */
133 0, /* minimum packet size */
134 INFPSZ, /* maximum packet size */
135 512, /* hi-water mark */
136 256 /* lo-water mark */
139 static struct qinit rloginmodrinit = {
140 rlmodrput,
141 rlmodrsrv,
142 rlmodopen,
143 rlmodclose,
144 nulldev,
145 &rloginmodiinfo,
146 NULL
149 static struct qinit rloginmodwinit = {
150 rlmodwput,
151 rlmodwsrv,
152 NULL,
153 NULL,
154 nulldev,
155 &rloginmodiinfo,
156 NULL
159 struct streamtab rloginmodinfo = {
160 &rloginmodrinit,
161 &rloginmodwinit,
162 NULL,
163 NULL
167 * Per-instance state struct for the rloginmod module.
169 struct rlmod_info
171 int flags;
172 bufcall_id_t wbufcid;
173 bufcall_id_t rbufcid;
174 timeout_id_t wtimoutid;
175 timeout_id_t rtimoutid;
176 int rl_expdat;
177 int stopmode;
178 mblk_t *unbind_mp;
179 char startc;
180 char stopc;
181 char oobdata[1];
182 mblk_t *wndw_sz_hd_mp;
186 * Flag used in flags
188 #define RL_DISABLED 0x1
189 #define RL_IOCPASSTHRU 0x2
191 /*ARGSUSED*/
192 static void
193 dummy_callback(void *arg)
197 * rlmodopen - open routine gets called when the
198 * module gets pushed onto the stream.
200 /*ARGSUSED*/
201 static int
202 rlmodopen(queue_t *q, dev_t *devp, int oflag, int sflag, cred_t *cred)
204 struct rlmod_info *rmip;
205 union T_primitives *tp;
206 mblk_t *bp;
207 int error;
209 if (sflag != MODOPEN)
210 return (EINVAL);
212 if (q->q_ptr != NULL) {
213 /* It's already attached. */
214 return (0);
218 * Allocate state structure.
220 rmip = kmem_zalloc(sizeof (*rmip), KM_SLEEP);
223 * Cross-link.
225 q->q_ptr = rmip;
226 WR(q)->q_ptr = rmip;
227 rmip->rl_expdat = 0;
228 rmip->stopmode = TIOCPKT_DOSTOP;
229 rmip->startc = CTRL('q');
230 rmip->stopc = CTRL('s');
231 rmip->oobdata[0] = (char)TIOCPKT_WINDOW;
232 rmip->wndw_sz_hd_mp = NULL;
234 * Allow only non-M_DATA blocks to pass up to in.rlogind until
235 * it is ready for M_DATA (indicated by RL_IOC_ENABLE).
237 rmip->flags |= RL_DISABLED;
239 qprocson(q);
242 * Since TCP operates in the TLI-inspired brain-dead fashion,
243 * the connection will revert to bound state if the connection
244 * is reset by the client. We must send a T_UNBIND_REQ in
245 * that case so the port doesn't get "wedged" (preventing
246 * inetd from being able to restart the listener). Allocate
247 * it here, so that we don't need to worry about allocb()
248 * failures later.
250 while ((rmip->unbind_mp = allocb(sizeof (union T_primitives),
251 BPRI_HI)) == NULL) {
252 bufcall_id_t id = qbufcall(q, sizeof (union T_primitives),
253 BPRI_HI, dummy_callback, NULL);
254 if (!qwait_sig(q)) {
255 qunbufcall(q, id);
256 error = EINTR;
257 goto fail;
259 qunbufcall(q, id);
261 rmip->unbind_mp->b_wptr = rmip->unbind_mp->b_rptr +
262 sizeof (struct T_unbind_req);
263 rmip->unbind_mp->b_datap->db_type = M_PROTO;
264 tp = (union T_primitives *)rmip->unbind_mp->b_rptr;
265 tp->type = T_UNBIND_REQ;
268 * Send a M_PROTO msg of type T_DATA_REQ (this is unique for
269 * read queue since only write queue can get T_DATA_REQ).
270 * Readstream routine in the daemon will do a getmsg() till
271 * it receives this proto message.
273 while ((bp = allocb(sizeof (union T_primitives), BPRI_HI)) == NULL) {
274 bufcall_id_t id = qbufcall(q, sizeof (union T_primitives),
275 BPRI_HI, dummy_callback, NULL);
276 if (!qwait_sig(q)) {
277 qunbufcall(q, id);
278 error = EINTR;
279 goto fail;
281 qunbufcall(q, id);
283 bp->b_datap->db_type = M_PROTO;
284 bp->b_wptr = bp->b_rptr + sizeof (union T_primitives);
285 tp = (union T_primitives *)bp->b_rptr;
286 tp->type = T_DATA_REQ;
287 tp->data_req.MORE_flag = 0;
289 putnext(q, bp);
290 return (0);
291 fail:
292 qprocsoff(q);
293 if (rmip->unbind_mp != NULL) {
294 freemsg(rmip->unbind_mp);
296 kmem_free(rmip, sizeof (struct rlmod_info));
297 q->q_ptr = NULL;
298 WR(q)->q_ptr = NULL;
299 return (error);
304 * rlmodclose - This routine gets called when the module
305 * gets popped off of the stream.
308 /*ARGSUSED*/
309 static int
310 rlmodclose(queue_t *q, int flag, cred_t *credp)
312 struct rlmod_info *rmip = (struct rlmod_info *)q->q_ptr;
313 mblk_t *mp;
316 * Flush any write-side data downstream. Ignoring flow
317 * control at this point is known to be safe because the
318 * M_HANGUP below poisons the stream such that no modules can
319 * be pushed again.
321 while (mp = getq(WR(q)))
322 putnext(WR(q), mp);
324 /* Poison the stream head so that we can't be pushed again. */
325 (void) putnextctl(q, M_HANGUP);
327 qprocsoff(q);
328 if (rmip->wbufcid) {
329 qunbufcall(q, rmip->wbufcid);
330 rmip->wbufcid = 0;
332 if (rmip->rbufcid) {
333 qunbufcall(q, rmip->rbufcid);
334 rmip->rbufcid = 0;
336 if (rmip->wtimoutid) {
337 (void) quntimeout(q, rmip->wtimoutid);
338 rmip->wtimoutid = 0;
340 if (rmip->rtimoutid) {
341 (void) quntimeout(q, rmip->rtimoutid);
342 rmip->rtimoutid = 0;
345 if (rmip->unbind_mp != NULL) {
346 freemsg(rmip->unbind_mp);
349 if (rmip->wndw_sz_hd_mp != NULL) {
350 freemsg(rmip->wndw_sz_hd_mp);
353 kmem_free(q->q_ptr, sizeof (struct rlmod_info));
354 q->q_ptr = WR(q)->q_ptr = NULL;
355 return (0);
359 * rlmodrput - Module read queue put procedure.
360 * This is called from the module or
361 * driver downstream.
364 static int
365 rlmodrput(queue_t *q, mblk_t *mp)
367 struct rlmod_info *rmip = (struct rlmod_info *)q->q_ptr;
368 union T_primitives *tip;
370 TRACE_2(TR_FAC_RLOGINP, TR_RLOGINP_RPUT_IN, "rlmodrput start: "
371 "q %p, mp %p", q, mp);
374 /* if low (normal) priority... */
375 if ((mp->b_datap->db_type < QPCTL) &&
376 /* ...and data is already queued... */
377 ((q->q_first) ||
378 /* ...or currently disabled and this is M_DATA... */
379 ((rmip->flags & RL_DISABLED) &&
380 (mp->b_datap->db_type == M_DATA)))) {
381 /* ...delay delivery of the message */
382 (void) putq(q, mp);
383 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_RPUT_OUT,
384 "rlmodrput end: q %p, mp %p, %s", q, mp, "flow");
385 return (0);
388 switch (mp->b_datap->db_type) {
390 case M_PROTO:
391 case M_PCPROTO:
392 tip = (union T_primitives *)mp->b_rptr;
393 switch (tip->type) {
395 case T_ORDREL_IND:
396 case T_DISCON_IND:
397 /* Make into M_HANGUP and putnext */
398 mp->b_datap->db_type = M_HANGUP;
399 mp->b_wptr = mp->b_rptr;
400 if (mp->b_cont) {
401 freemsg(mp->b_cont);
402 mp->b_cont = NULL;
405 * If we haven't already, send T_UNBIND_REQ to prevent
406 * TCP from going into "BOUND" state and locking up the
407 * port.
409 if (tip->type == T_DISCON_IND && rmip->unbind_mp !=
410 NULL) {
411 putnext(q, mp);
412 qreply(q, rmip->unbind_mp);
413 rmip->unbind_mp = NULL;
414 } else {
415 putnext(q, mp);
417 break;
420 * We only get T_OK_ACK when we issue the unbind, and it can
421 * be ignored safely.
423 case T_OK_ACK:
424 ASSERT(rmip->unbind_mp == NULL);
425 freemsg(mp);
426 break;
428 default:
429 cmn_err(CE_NOTE,
430 "rlmodrput: got 0x%x type M_PROTO/M_PCPROTO msg",
431 tip->type);
432 freemsg(mp);
434 break;
436 case M_DATA:
437 if (canputnext(q) && q->q_first == NULL) {
438 (void) rlmodrmsg(q, mp);
439 } else {
440 (void) putq(q, mp);
442 break;
444 case M_FLUSH:
446 * Since M_FLUSH came from TCP, we mark it bound for
447 * daemon, not tty. This only happens when TCP expects
448 * to do a connection reset.
450 mp->b_flag |= MSGMARK;
451 if (*mp->b_rptr & FLUSHR)
452 flushq(q, FLUSHALL);
454 putnext(q, mp);
455 break;
457 case M_PCSIG:
458 case M_ERROR:
459 case M_IOCACK:
460 case M_IOCNAK:
461 case M_SETOPTS:
462 if (mp->b_datap->db_type <= QPCTL && !canputnext(q))
463 (void) putq(q, mp);
464 else
465 putnext(q, mp);
466 break;
468 default:
469 #ifdef DEBUG
470 cmn_err(CE_NOTE, "rlmodrput: unexpected msg type 0x%x",
471 mp->b_datap->db_type);
472 #endif
473 freemsg(mp);
475 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_RPUT_OUT, "rlmodrput end: q %p, "
476 "mp %p, %s", q, mp, "done");
477 return (0);
481 * rlmodrsrv - module read service procedure
483 static int
484 rlmodrsrv(queue_t *q)
486 mblk_t *mp;
487 struct rlmod_info *rmip = (struct rlmod_info *)q->q_ptr;
488 union T_primitives *tip;
490 TRACE_1(TR_FAC_RLOGINP, TR_RLOGINP_RSRV_IN, "rlmodrsrv start: "
491 "q %p", q);
492 while ((mp = getq(q)) != NULL) {
494 switch (mp->b_datap->db_type) {
495 case M_DATA:
496 if (rmip->flags & RL_DISABLED) {
497 (void) putbq(q, mp);
498 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_RSRV_OUT,
499 "rlmodrsrv end: q %p, mp %p, %s", q, mp,
500 "disabled");
501 return (0);
503 if (!canputnext(q)) {
504 (void) putbq(q, mp);
505 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_RSRV_OUT,
506 "rlmodrsrv end: q %p, mp %p, %s",
507 q, mp, "!canputnext");
508 return (0);
510 if (!rlmodrmsg(q, mp)) {
511 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_RSRV_OUT,
512 "rlmodrsrv end: q %p, mp %p, %s",
513 q, mp, "!rlmodrmsg");
514 return (0);
516 break;
518 case M_PROTO:
519 tip = (union T_primitives *)mp->b_rptr;
520 switch (tip->type) {
522 case T_ORDREL_IND:
523 case T_DISCON_IND:
524 /* Make into M_HANGUP and putnext */
525 mp->b_datap->db_type = M_HANGUP;
526 mp->b_wptr = mp->b_rptr;
527 if (mp->b_cont) {
528 freemsg(mp->b_cont);
529 mp->b_cont = NULL;
532 * If we haven't already, send T_UNBIND_REQ
533 * to prevent TCP from going into "BOUND"
534 * state and locking up the port.
536 if (tip->type == T_DISCON_IND &&
537 rmip->unbind_mp != NULL) {
538 putnext(q, mp);
539 qreply(q, rmip->unbind_mp);
540 rmip->unbind_mp = NULL;
541 } else {
542 putnext(q, mp);
544 break;
547 * We only get T_OK_ACK when we issue the unbind, and
548 * it can be ignored safely.
550 case T_OK_ACK:
551 ASSERT(rmip->unbind_mp == NULL);
552 freemsg(mp);
553 break;
555 default:
556 cmn_err(CE_NOTE,
557 "rlmodrsrv: got 0x%x type PROTO msg",
558 tip->type);
559 freemsg(mp);
561 break;
563 case M_SETOPTS:
564 if (!canputnext(q)) {
565 (void) putbq(q, mp);
566 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_RSRV_OUT,
567 "rlmodrsrv end: q %p, mp %p, %s",
568 q, mp, "!canputnext M_SETOPTS");
569 return (0);
571 putnext(q, mp);
572 break;
574 default:
575 #ifdef DEBUG
576 cmn_err(CE_NOTE,
577 "rlmodrsrv: unexpected msg type 0x%x",
578 mp->b_datap->db_type);
579 #endif
580 freemsg(mp);
584 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_RSRV_OUT, "rlmodrsrv end: q %p, "
585 "mp %p, %s", q, mp, "empty");
587 return (0);
591 * rlmodwput - Module write queue put procedure.
592 * All non-zero messages are send downstream unchanged
594 static int
595 rlmodwput(queue_t *q, mblk_t *mp)
597 char cntl;
598 struct rlmod_info *rmip = (struct rlmod_info *)q->q_ptr;
599 mblk_t *tmpmp;
600 int rw;
602 TRACE_2(TR_FAC_RLOGINP, TR_RLOGINP_WPUT_IN, "rlmodwput start: "
603 "q %p, mp %p", q, mp);
605 if (rmip->rl_expdat) {
607 * call make_expmblk to create an expedited
608 * message block.
610 cntl = rmip->oobdata[0] | TIOCPKT_FLUSHWRITE;
612 if (!canputnext(q)) {
613 (void) putq(q, mp);
614 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_WPUT_OUT,
615 "rlmodwput end: q %p, mp %p, %s",
616 q, mp, "expdata && !canputnext");
617 return (0);
619 if ((tmpmp = make_expmblk(cntl))) {
620 putnext(q, tmpmp);
621 rmip->rl_expdat = 0;
622 } else {
623 recover1(q, sizeof (mblk_t)); /* XXX.sparker */
627 if ((q->q_first || rmip->rl_expdat) && mp->b_datap->db_type < QPCTL) {
628 (void) putq(q, mp);
629 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_WPUT_OUT, "rlmodwput end: "
630 "q %p, mp %p, %s", q, mp, "queued data");
631 return (0);
633 switch (mp->b_datap->db_type) {
635 case M_DATA:
636 if (!canputnext(q))
637 (void) putq(q, mp);
638 else
639 putnext(q, mp);
640 break;
642 case M_FLUSH:
644 * We must take care to create and forward out-of-band data
645 * indicating the flush to the far side.
647 rw = *mp->b_rptr;
648 *mp->b_rptr &= ~FLUSHW;
649 qreply(q, mp);
650 if (rw & FLUSHW) {
652 * Since all rlogin protocol data is sent in this
653 * direction as urgent data, and TCP does not flush
654 * urgent data, it is okay to actually forward this
655 * flush. (telmod cannot.)
657 flushq(q, FLUSHDATA);
659 * The putnextctl1() call can only fail if we're
660 * out of memory. Ideally, we might set a state
661 * bit and reschedule ourselves when memory
662 * becomes available, so we make sure not to miss
663 * sending the FLUSHW to TCP before the urgent
664 * byte. Not doing this just means in some cases
665 * a bit more trash passes before the flush takes
666 * hold.
668 (void) putnextctl1(q, M_FLUSH, FLUSHW);
670 * Notify peer of the write flush request.
672 cntl = rmip->oobdata[0] | TIOCPKT_FLUSHWRITE;
673 if (!canputnext(q)) {
674 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_WPUT_OUT,
675 "rlmodwput end: q %p, mp %p, %s",
676 q, mp, "flushw && !canputnext");
677 return (0);
679 if ((mp = make_expmblk(cntl)) == NULL) {
680 rmip->rl_expdat = 1;
681 recover1(q, sizeof (mblk_t));
682 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_WPUT_OUT,
683 "rlmodwput end: q %p, mp %p, %s",
684 q, mp, "!make_expmblk");
685 return (0);
687 putnext(q, mp);
689 break;
691 case M_IOCTL:
692 if (!rlmodwioctl(q, mp))
693 (void) putq(q, mp);
694 break;
696 case M_PROTO:
697 switch (((union T_primitives *)mp->b_rptr)->type) {
698 case T_EXDATA_REQ:
699 case T_ORDREL_REQ:
700 case T_DISCON_REQ:
701 putnext(q, mp);
702 break;
704 default:
705 #ifdef DEBUG
706 cmn_err(CE_NOTE,
707 "rlmodwput: unexpected TPI primitive 0x%x",
708 ((union T_primitives *)mp->b_rptr)->type);
709 #endif
710 freemsg(mp);
712 break;
714 case M_PCPROTO:
715 if (((struct T_exdata_req *)mp->b_rptr)->PRIM_type ==
716 T_DISCON_REQ) {
717 putnext(q, mp);
718 } else {
719 /* XXX.sparker Log unexpected message */
720 freemsg(mp);
722 break;
724 default:
725 #ifdef DEBUG
726 cmn_err(CE_NOTE,
727 "rlmodwput: unexpected msg type 0x%x",
728 mp->b_datap->db_type);
729 #endif
730 freemsg(mp);
731 break;
733 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_WPUT_OUT, "rlmodwput end: "
734 "q %p, mp %p, %s", q, mp, "done");
735 return (0);
739 * rlmodwsrv - module write service procedure
741 static int
742 rlmodwsrv(queue_t *q)
744 mblk_t *mp, *tmpmp;
745 char cntl;
746 struct rlmod_info *rmip = (struct rlmod_info *)q->q_ptr;
748 TRACE_1(TR_FAC_RLOGINP, TR_RLOGINP_WSRV_IN, "rlmodwsrv "
749 "start: q %p", q);
750 if (rmip->rl_expdat) {
752 * call make_expmblk to create an expedited
753 * message block.
755 cntl = rmip->oobdata[0] | TIOCPKT_FLUSHWRITE;
756 if (!canputnext(q)) {
757 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_WSRV_OUT,
758 "rlmodwsrv end: q %p, mp %p, %s",
759 q, NULL, "!canputnext && expdat");
760 return (0);
762 if ((tmpmp = make_expmblk(cntl))) {
763 putnext(q, tmpmp);
764 rmip->rl_expdat = 0;
765 } else {
766 recover1(q, sizeof (mblk_t));
767 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_WSRV_OUT,
768 "rlmodwsrv end: q %p, mp %p, %s",
769 q, NULL, "!make_expmblk");
770 return (0);
773 while ((mp = getq(q)) != NULL) {
775 if (!canputnext(q) || rmip->rl_expdat) {
776 (void) putbq(q, mp);
777 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_WSRV_OUT,
778 "rlmodwsrv end: q %p, mp %p, %s",
779 q, mp, "!canputnext || expdat");
780 return (0);
782 if (mp->b_datap->db_type == M_IOCTL) {
783 if (!rlmodwioctl(q, mp)) {
784 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_WSRV_OUT,
785 "rlmodwsrv end: q %p, mp %p, %s",
786 q, mp, "!rlmodwioctl");
787 (void) putbq(q, mp);
788 return (0);
790 continue;
792 putnext(q, mp);
794 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_WSRV_OUT, "rlmodwsrv end: q %p, "
795 "mp %p, %s", q, mp, "done");
796 return (0);
800 * This routine returns a message block with an expedited
801 * data request
803 static mblk_t *
804 make_expmblk(char cntl)
806 mblk_t *mp;
807 mblk_t *bp;
808 struct T_exdata_req *data_req;
810 bp = allocb(sizeof (struct T_exdata_req), BPRI_MED);
811 if (bp == NULL)
812 return (NULL);
813 if ((mp = allocb(sizeof (char), BPRI_MED)) == NULL) {
814 freeb(bp);
815 return (NULL);
817 bp->b_datap->db_type = M_PROTO;
818 data_req = (struct T_exdata_req *)bp->b_rptr;
819 data_req->PRIM_type = T_EXDATA_REQ;
820 data_req->MORE_flag = 0;
822 bp->b_wptr += sizeof (struct T_exdata_req);
824 * Send a 1 byte data message block with appropriate
825 * control character.
827 mp->b_datap->db_type = M_DATA;
828 mp->b_wptr = mp->b_rptr + 1;
829 (*(char *)(mp->b_rptr)) = cntl;
830 bp->b_cont = mp;
831 return (bp);
834 * This routine parses M_DATA messages checking for window size protocol
835 * from a given message block. It returns TRUE if no resource exhaustion
836 * conditions are found. This is for use in the service procedure, which
837 * needs to know whether to continue, or stop processing the queue.
839 static int
840 rlmodrmsg(queue_t *q, mblk_t *mp)
842 unsigned char *tmp, *tmp1;
843 mblk_t *newmp;
844 size_t sz;
845 ssize_t count, newcount = 0;
846 struct rlmod_info *rmip = (struct rlmod_info *)q->q_ptr;
849 * Eliminate any zero length messages here, so we don't filter EOFs
850 * accidentally.
852 if (msgdsize(mp) == 0) {
853 ASSERT(rmip->wndw_sz_hd_mp == NULL);
854 goto out;
857 * Check if we have stored a previous message block because a window
858 * update was split over TCP segments. If so, append the new one to
859 * the stored one and process the stored one as if it just arrived.
861 if (rmip->wndw_sz_hd_mp != NULL) {
862 linkb(rmip->wndw_sz_hd_mp, mp);
863 mp = rmip->wndw_sz_hd_mp;
864 rmip->wndw_sz_hd_mp = NULL;
866 newmp = mp;
868 while (mp) {
869 tmp = mp->b_rptr;
871 * scan through the entire message block
873 while (tmp < mp->b_wptr) {
875 * check for FF (rlogin magic escape sequence)
877 if (tmp[0] == RLOGIN_MAGIC) {
879 * Update bytes read so far.
881 count = newcount + tmp - mp->b_rptr;
883 * Pull together message chain in case
884 * window escape is split across blocks.
886 if ((pullupmsg(newmp, -1)) == 0) {
887 sz = msgdsize(newmp);
888 recover(q, newmp, sz);
889 return (0);
892 * pullupmsg results in newmp consuming
893 * all message blocks in this chain, and
894 * therefor mp wants updating.
896 mp = newmp;
899 * adjust tmp to where we
900 * stopped - count keeps track
901 * of bytes read so far.
902 * reset newcount = 0.
904 tmp = mp->b_rptr + count;
905 newcount = 0;
908 * Use the variable tmp1 to compute where
909 * the end of the window escape (currently
910 * the only rlogin protocol sequence), then
911 * check to see if we got all those bytes.
913 tmp1 = tmp + 4 + sizeof (struct winsize);
915 if (tmp1 > mp->b_wptr) {
917 * All the window escape bytes aren't
918 * in this TCP segment. Store this
919 * mblk to one side so we can append
920 * the rest of the escape to it when
921 * its segment arrives.
923 rmip->wndw_sz_hd_mp = mp;
924 return (TRUE);
927 * check for FF FF s s pattern
929 if ((tmp[1] == RLOGIN_MAGIC) &&
930 (tmp[2] == 's') && (tmp[3] == 's')) {
933 * If rlwinsetup returns an error,
934 * we do recover with newmp which
935 * points to new chain of mblks after
936 * doing window control ioctls.
937 * rlwinsetup returns newmp which
938 * contains only data part.
939 * Note that buried inside rlwinsetup
940 * is where we do the putnext.
942 if (rlwinsetup(q, mp, tmp) == NULL) {
943 sz = msgdsize(mp);
944 recover(q, mp, sz);
945 return (0);
948 * We have successfully consumed the
949 * window sequence, but rlwinsetup()
950 * and its children have moved memory
951 * up underneath us. This means that
952 * the byte underneath *tmp has not
953 * been scanned now. We will now need
954 * to rescan it.
956 continue;
959 tmp++;
962 * bump newcount to include size of this particular block.
964 newcount += (mp->b_wptr - mp->b_rptr);
965 mp = mp->b_cont;
968 * If we trimmed the message down to nothing to forward, don't
969 * send any M_DATA message. (Don't want to send EOF!)
971 if (msgdsize(newmp) == 0) {
972 freemsg(newmp);
973 newmp = NULL;
975 out:
976 if (newmp) {
977 if (!canputnext(q)) {
978 (void) putbq(q, newmp);
979 return (0);
980 } else {
981 putnext(q, newmp);
984 return (TRUE);
989 * This routine is called to handle window size changes.
990 * The routine returns 1 on success and 0 on error (allocb failure).
992 static int
993 rlwinctl(queue_t *q, mblk_t *mp)
995 mblk_t *rl_msgp;
996 struct iocblk *iocbp;
997 struct rlmod_info *rmip = (struct rlmod_info *)q->q_ptr;
999 TRACE_2(TR_FAC_RLOGINP, TR_RLOGINP_WINCTL_IN, "rlwinctl start: q %p, "
1000 "mp %p", q, mp);
1002 rmip->oobdata[0] &= ~TIOCPKT_WINDOW; /* we know they heard */
1004 if ((rl_msgp = mkiocb(TIOCSWINSZ)) == NULL) {
1005 TRACE_2(TR_FAC_RLOGINP, TR_RLOGINP_WINCTL_OUT, "rlwinctl end: "
1006 "q %p, mp %p, allocb failed", q, mp);
1007 return (0);
1011 * create an M_IOCTL message type.
1013 rl_msgp->b_cont = mp;
1014 iocbp = (struct iocblk *)rl_msgp->b_rptr;
1015 iocbp->ioc_count = msgdsize(mp);
1017 putnext(q, rl_msgp);
1018 TRACE_2(TR_FAC_RLOGINP, TR_RLOGINP_WINCTL_OUT, "rlwinctl end: "
1019 "q %p, mp %p, done", q, mp);
1020 return (1);
1024 * This routine sets up window size change protocol.
1025 * The routine returns the new mblk after issuing rlwinctl
1026 * for window size changes. New mblk contains only data part
1027 * of the message block. The routine returns 0 on error.
1029 static mblk_t *
1030 rlwinsetup(queue_t *q, mblk_t *mp, unsigned char *blk)
1032 mblk_t *mp1;
1033 unsigned char *jmpmp;
1034 ssize_t left = 0;
1035 struct winsize win;
1038 * Set jmpmp to where to jump, to get just past the end of the
1039 * window size protocol sequence.
1041 jmpmp = (blk + 4 + sizeof (struct winsize));
1042 left = mp->b_wptr - jmpmp;
1044 if ((mp1 = allocb(sizeof (struct winsize), BPRI_MED)) == NULL)
1045 return (0);
1046 mp1->b_datap->db_type = M_DATA;
1047 mp1->b_wptr = mp1->b_rptr + sizeof (struct winsize);
1048 bcopy(blk + 4, &win, sizeof (struct winsize));
1049 win.ws_row = ntohs(win.ws_row);
1050 win.ws_col = ntohs(win.ws_col);
1051 win.ws_xpixel = ntohs(win.ws_xpixel);
1052 win.ws_ypixel = ntohs(win.ws_ypixel);
1053 bcopy(&win, mp1->b_rptr, sizeof (struct winsize));
1055 if ((rlwinctl(q, mp1)) == 0) {
1056 freeb(mp1);
1057 return (0);
1059 if (left > 0) {
1061 * Must delete the window size protocol sequence. We do
1062 * this by sliding all the stuff after the sequence (jmpmp)
1063 * to where the sequence itself began (blk).
1065 bcopy(jmpmp, blk, left);
1066 mp->b_wptr = blk + left;
1067 } else
1068 mp->b_wptr = blk;
1069 return (mp);
1073 * When an ioctl changes software flow control on the tty, we must notify
1074 * the rlogin client, so it can adjust its behavior appropriately. This
1075 * routine, called from either the put or service routine, determines if
1076 * the flow handling has changed. If so, it tries to send the indication
1077 * to the client. It returns true or false depending upon whether the
1078 * message was fully processed. If it wasn't fully processed it queues
1079 * the message for retry later when resources
1080 * (allocb/canputnext) are available.
1082 static boolean_t
1083 tty_flow(queue_t *q, struct rlmod_info *rmip, mblk_t *mp)
1085 struct iocblk *ioc;
1086 struct termios *tp;
1087 struct termio *ti;
1088 int stop, ixon;
1089 mblk_t *tmpmp;
1090 char cntl;
1091 int error;
1093 ioc = (struct iocblk *)mp->b_rptr;
1094 switch (ioc->ioc_cmd) {
1097 * If it is a tty ioctl, save the output flow
1098 * control flag and the start and stop flow control
1099 * characters if they are available.
1101 case TCSETS:
1102 case TCSETSW:
1103 case TCSETSF:
1104 error = miocpullup(mp, sizeof (struct termios));
1105 if (error != 0) {
1106 miocnak(q, mp, 0, error);
1107 return (B_TRUE);
1109 tp = (struct termios *)(mp->b_cont->b_rptr);
1110 rmip->stopc = tp->c_cc[VSTOP];
1111 rmip->startc = tp->c_cc[VSTART];
1112 ixon = tp->c_iflag & IXON;
1113 break;
1115 case TCSETA:
1116 case TCSETAW:
1117 case TCSETAF:
1118 error = miocpullup(mp, sizeof (struct termio));
1119 if (error != 0) {
1120 miocnak(q, mp, 0, error);
1121 return (B_TRUE);
1123 ti = (struct termio *)(mp->b_cont->b_rptr);
1124 ixon = ti->c_iflag & IXON;
1125 break;
1127 default:
1129 * This function must never be called for an M_IOCTL
1130 * except the listed ones.
1132 #ifdef DEBUG
1133 cmn_err(CE_PANIC,
1134 "rloginmod: tty_flow: bad ioctl 0x%x", ioc->ioc_cmd);
1135 #else
1136 miocnak(q, mp, 0, EINVAL);
1137 return (B_TRUE);
1138 #endif
1141 * If tty ioctl processing is done, check for stopmode
1143 stop = (ixon && (rmip->stopc == CTRL('s')) &&
1144 (rmip->startc == CTRL('q')));
1145 if (rmip->stopmode == TIOCPKT_NOSTOP) {
1146 if (stop) {
1147 cntl = rmip->oobdata[0] | TIOCPKT_DOSTOP;
1148 if ((tmpmp = make_expmblk(cntl)) == NULL) {
1149 recover(q, mp, sizeof (mblk_t));
1150 return (B_FALSE);
1152 if (!canputnext(q)) {
1153 freemsg(tmpmp);
1154 return (B_FALSE);
1156 putnext(q, tmpmp);
1157 rmip->stopmode = TIOCPKT_DOSTOP;
1159 } else {
1160 if (!stop) {
1161 cntl = rmip->oobdata[0] | TIOCPKT_NOSTOP;
1162 if ((tmpmp = make_expmblk(cntl)) == NULL) {
1163 recover(q, mp, sizeof (mblk_t));
1164 return (B_FALSE);
1166 if (!canputnext(q)) {
1167 freemsg(tmpmp);
1168 return (B_FALSE);
1170 putnext(q, tmpmp);
1171 rmip->stopmode = TIOCPKT_NOSTOP;
1175 miocack(q, mp, 0, 0);
1176 return (B_TRUE);
1179 /* rlmodwioctl - handle M_IOCTL messages on the write queue. */
1181 static boolean_t
1182 rlmodwioctl(queue_t *q, mblk_t *mp)
1184 struct iocblk *ioc;
1185 struct rlmod_info *rmip = (struct rlmod_info *)q->q_ptr;
1186 int error;
1188 ioc = (struct iocblk *)mp->b_rptr;
1189 switch (ioc->ioc_cmd) {
1192 * This is a special ioctl to reenable the queue.
1193 * The initial data read from the stream head is
1194 * put back on the queue.
1196 case RL_IOC_ENABLE:
1198 * Send negative ack if RL_DISABLED flag is not set
1201 if (!(rmip->flags & RL_DISABLED)) {
1202 miocnak(q, mp, 0, EINVAL);
1203 break;
1205 if (mp->b_cont) {
1206 (void) putbq(RD(q), mp->b_cont);
1207 mp->b_cont = NULL;
1210 if (rmip->flags & RL_DISABLED)
1211 rmip->flags &= ~RL_DISABLED;
1212 qenable(RD(q));
1213 miocack(q, mp, 0, 0);
1214 TRACE_3(TR_FAC_RLOGINP, TR_RLOGINP_WPUT_OUT,
1215 "rlmodwput end: q %p, mp %p, %s",
1216 q, mp, "IOCACK enable");
1217 return (B_TRUE);
1220 * If it is a tty ioctl, save the output flow
1221 * control flag and the start and stop flow control
1222 * characters if they are available.
1224 case TCSETS:
1225 case TCSETSW:
1226 case TCSETSF:
1227 case TCSETA:
1228 case TCSETAW:
1229 case TCSETAF:
1230 return (tty_flow(q, rmip, mp));
1232 #ifdef DEBUG
1233 case TIOCSWINSZ:
1234 case TIOCSTI:
1235 case TCSBRK:
1236 miocnak(q, mp, 0, EINVAL);
1237 break;
1238 #endif
1239 case CRYPTPASSTHRU:
1240 error = miocpullup(mp, sizeof (uchar_t));
1241 if (error != 0) {
1242 miocnak(q, mp, 0, error);
1243 break;
1245 if (*(mp->b_cont->b_rptr) == 0x01)
1246 rmip->flags |= RL_IOCPASSTHRU;
1247 else
1248 rmip->flags &= ~RL_IOCPASSTHRU;
1250 miocack(q, mp, 0, 0);
1251 break;
1253 default:
1254 if (rmip->flags & RL_IOCPASSTHRU) {
1255 putnext(q, mp);
1256 } else {
1257 #ifdef DEBUG
1258 cmn_err(CE_NOTE,
1259 "rlmodwioctl: unexpected ioctl type 0x%x",
1260 ioc->ioc_cmd);
1261 #endif
1262 miocnak(q, mp, 0, EINVAL);
1265 return (B_TRUE);
1268 static void
1269 rlmod_timer(void *arg)
1271 queue_t *q = arg;
1272 struct rlmod_info *rmip = (struct rlmod_info *)q->q_ptr;
1274 ASSERT(rmip);
1275 if (q->q_flag & QREADR) {
1276 ASSERT(rmip->rtimoutid);
1277 rmip->rtimoutid = 0;
1278 } else {
1279 ASSERT(rmip->wtimoutid);
1280 rmip->wtimoutid = 0;
1282 enableok(q);
1283 qenable(q);
1286 static void
1287 rlmod_buffer(void *arg)
1289 queue_t *q = arg;
1290 struct rlmod_info *rmip = (struct rlmod_info *)q->q_ptr;
1292 ASSERT(rmip);
1293 if (q->q_flag & QREADR) {
1294 ASSERT(rmip->rbufcid);
1295 rmip->rbufcid = 0;
1296 } else {
1297 ASSERT(rmip->wbufcid);
1298 rmip->wbufcid = 0;
1300 enableok(q);
1301 qenable(q);
1304 static void
1305 recover(queue_t *q, mblk_t *mp, size_t size)
1308 * Avoid re-enabling the queue.
1310 ASSERT(mp->b_datap->db_type < QPCTL);
1312 noenable(q);
1313 (void) putbq(q, mp);
1314 recover1(q, size);
1317 static void
1318 recover1(queue_t *q, size_t size)
1320 struct rlmod_info *rmip = (struct rlmod_info *)q->q_ptr;
1321 timeout_id_t tid;
1322 bufcall_id_t bid;
1325 * Make sure there is at most one outstanding request per queue.
1327 if (q->q_flag & QREADR) {
1328 if (rmip->rtimoutid || rmip->rbufcid)
1329 return;
1330 } else {
1331 if (rmip->wtimoutid || rmip->wbufcid)
1332 return;
1334 if (!(bid = qbufcall(RD(q), size, BPRI_MED, rlmod_buffer, q))) {
1335 tid = qtimeout(RD(q), rlmod_timer, q, SIMWAIT);
1336 if (q->q_flag & QREADR)
1337 rmip->rtimoutid = tid;
1338 else
1339 rmip->wtimoutid = tid;
1340 } else {
1341 if (q->q_flag & QREADR)
1342 rmip->rbufcid = bid;
1343 else
1344 rmip->wbufcid = bid;