123390Smckusick /*
263690Sbostic * Copyright (c) 1982, 1986, 1989, 1993
363690Sbostic * The Regents of the University of California. All rights reserved.
423390Smckusick *
544446Sbostic * %sccs.include.redist.c%
638007Sbostic *
7*68389Sdab * @(#)tty_pty.c 8.4 (Berkeley) 02/20/95
823390Smckusick */
92283Stoy
102281Stoy /*
112281Stoy * Pseudo-teletype Driver
122281Stoy * (Actually two drivers, requiring two entries in 'cdevsw')
132281Stoy */
1459119Storek #include "pty.h" /* XXX */
152314Stoy
1656517Sbostic #include <sys/param.h>
1756517Sbostic #include <sys/systm.h>
1856517Sbostic #include <sys/ioctl.h>
1956517Sbostic #include <sys/proc.h>
2056517Sbostic #include <sys/tty.h>
2156517Sbostic #include <sys/conf.h>
2256517Sbostic #include <sys/file.h>
2356517Sbostic #include <sys/uio.h>
2456517Sbostic #include <sys/kernel.h>
2556517Sbostic #include <sys/vnode.h>
266239Sroot
277475Ssam #if NPTY == 1
2818651Sbloom #undef NPTY
296239Sroot #define NPTY 32 /* crude XXX */
307475Ssam #endif
312281Stoy
3216788Ssam #define BUFSIZ 100 /* Chunk size iomoved to/from user */
334484Swnj
342281Stoy /*
3518651Sbloom * pts == /dev/tty[pqrs]?
3618651Sbloom * ptc == /dev/pty[pqrs]?
372281Stoy */
3859119Storek struct tty pt_tty[NPTY]; /* XXX */
394484Swnj struct pt_ioctl {
405427Swnj int pt_flags;
4152523Smckusick struct selinfo pt_selr, pt_selw;
4218651Sbloom u_char pt_send;
4318651Sbloom u_char pt_ucntl;
4459119Storek } pt_ioctl[NPTY]; /* XXX */
4518651Sbloom int npty = NPTY; /* for pstat -t */
462281Stoy
4735811Smarc #define PF_PKT 0x08 /* packet mode */
4835811Smarc #define PF_STOPPED 0x10 /* user told stopped */
4935811Smarc #define PF_REMOTE 0x20 /* remote and flow controlled input */
5035811Smarc #define PF_NOSTOP 0x40
5135811Smarc #define PF_UCNTL 0x80 /* user control mode */
522281Stoy
5352411Storek void ptsstop __P((struct tty *, int));
5452411Storek
5559119Storek /*
5659119Storek * Establish n (or default if n is 1) ptys in the system.
5759119Storek *
5859119Storek * XXX cdevsw & pstat require the array `pty[]' to be an array
5959119Storek */
6059119Storek void
ptyattach(n)6159119Storek ptyattach(n)
6259119Storek int n;
6359119Storek {
6459119Storek #ifdef notyet
6559119Storek char *mem;
6659119Storek register u_long ntb;
6759119Storek #define DEFAULT_NPTY 32
6859119Storek
6959119Storek /* maybe should allow 0 => none? */
7059119Storek if (n <= 1)
7159119Storek n = DEFAULT_NPTY;
7259119Storek ntb = n * sizeof(struct tty);
7359119Storek mem = malloc(ntb + ALIGNBYTES + n * sizeof(struct pt_ioctl),
7459119Storek M_DEVBUF, M_WAITOK);
7559119Storek pt_tty = (struct tty *)mem;
7659119Storek mem = (char *)ALIGN(mem + ntb);
7759119Storek pt_ioctl = (struct pt_ioctl *)mem;
7859119Storek npty = n;
7959119Storek #endif
8059119Storek }
8159119Storek
822281Stoy /*ARGSUSED*/
ptsopen(dev,flag,devtype,p)8348442Skarels ptsopen(dev, flag, devtype, p)
845396Sroot dev_t dev;
8552411Storek int flag, devtype;
8648442Skarels struct proc *p;
874484Swnj {
882281Stoy register struct tty *tp;
8918651Sbloom int error;
902281Stoy
9159119Storek if (minor(dev) >= npty)
928563Sroot return (ENXIO);
932281Stoy tp = &pt_tty[minor(dev)];
945408Swnj if ((tp->t_state & TS_ISOPEN) == 0) {
9544312Smarc tp->t_state |= TS_WOPEN;
962427Swnj ttychars(tp); /* Set up default chars */
9735811Smarc tp->t_iflag = TTYDEF_IFLAG;
9835811Smarc tp->t_oflag = TTYDEF_OFLAG;
9935811Smarc tp->t_lflag = TTYDEF_LFLAG;
10035811Smarc tp->t_cflag = TTYDEF_CFLAG;
10135811Smarc tp->t_ispeed = tp->t_ospeed = TTYDEF_SPEED;
10235811Smarc ttsetwater(tp); /* would be done in xxparam() */
10347548Skarels } else if (tp->t_state&TS_XCLUDE && p->p_ucred->cr_uid != 0)
1048563Sroot return (EBUSY);
1054484Swnj if (tp->t_oproc) /* Ctrlr still around. */
1065408Swnj tp->t_state |= TS_CARR_ON;
1075408Swnj while ((tp->t_state & TS_CARR_ON) == 0) {
1085408Swnj tp->t_state |= TS_WOPEN;
10949941Smckusick if (flag&FNONBLOCK)
11037590Smarc break;
11144312Smarc if (error = ttysleep(tp, (caddr_t)&tp->t_rawq, TTIPRI | PCATCH,
11240714Skarels ttopen, 0))
11340714Skarels return (error);
1142281Stoy }
11552411Storek error = (*linesw[tp->t_line].l_open)(dev, tp);
11618651Sbloom ptcwakeup(tp, FREAD|FWRITE);
11718651Sbloom return (error);
1182281Stoy }
1192281Stoy
ptsclose(dev,flag,mode,p)12049753Smarc ptsclose(dev, flag, mode, p)
1215396Sroot dev_t dev;
12249753Smarc int flag, mode;
12349753Smarc struct proc *p;
1245408Swnj {
1252281Stoy register struct tty *tp;
12653645Smarc int err;
1272281Stoy
1282281Stoy tp = &pt_tty[minor(dev)];
12953645Smarc err = (*linesw[tp->t_line].l_close)(tp, flag);
13053645Smarc err |= ttyclose(tp);
13118651Sbloom ptcwakeup(tp, FREAD|FWRITE);
13253645Smarc return (err);
1332281Stoy }
1342281Stoy
ptsread(dev,uio,flag)13537590Smarc ptsread(dev, uio, flag)
1365396Sroot dev_t dev;
1377823Sroot struct uio *uio;
13852411Storek int flag;
1394484Swnj {
14047548Skarels struct proc *p = curproc;
1415894Swnj register struct tty *tp = &pt_tty[minor(dev)];
1425894Swnj register struct pt_ioctl *pti = &pt_ioctl[minor(dev)];
1438521Sroot int error = 0;
1442281Stoy
1455894Swnj again:
1465894Swnj if (pti->pt_flags & PF_REMOTE) {
14747548Skarels while (isbackground(p, tp)) {
14847548Skarels if ((p->p_sigignore & sigmask(SIGTTIN)) ||
14947548Skarels (p->p_sigmask & sigmask(SIGTTIN)) ||
15047548Skarels p->p_pgrp->pg_jobc == 0 ||
15164576Sbostic p->p_flag & P_PPWAIT)
1528521Sroot return (EIO);
15347548Skarels pgsignal(p->p_pgrp, SIGTTIN, 1);
15444312Smarc if (error = ttysleep(tp, (caddr_t)&lbolt,
15544312Smarc TTIPRI | PCATCH, ttybg, 0))
15640714Skarels return (error);
1575408Swnj }
15818651Sbloom if (tp->t_canq.c_cc == 0) {
15937728Smckusick if (flag & IO_NDELAY)
1608521Sroot return (EWOULDBLOCK);
16144312Smarc if (error = ttysleep(tp, (caddr_t)&tp->t_canq,
16240714Skarels TTIPRI | PCATCH, ttyin, 0))
16340714Skarels return (error);
1645894Swnj goto again;
1655894Swnj }
16618651Sbloom while (tp->t_canq.c_cc > 1 && uio->uio_resid > 0)
16718651Sbloom if (ureadc(getc(&tp->t_canq), uio) < 0) {
1688521Sroot error = EFAULT;
1697823Sroot break;
1707823Sroot }
17118651Sbloom if (tp->t_canq.c_cc == 1)
17218651Sbloom (void) getc(&tp->t_canq);
17318651Sbloom if (tp->t_canq.c_cc)
1748521Sroot return (error);
1755894Swnj } else
1765894Swnj if (tp->t_oproc)
17737590Smarc error = (*linesw[tp->t_line].l_read)(tp, uio, flag);
17818651Sbloom ptcwakeup(tp, FWRITE);
1798521Sroot return (error);
1802281Stoy }
1812281Stoy
1825408Swnj /*
1835408Swnj * Write to pseudo-tty.
1845408Swnj * Wakeups of controlling tty will happen
1855408Swnj * indirectly, when tty driver calls ptsstart.
1865408Swnj */
ptswrite(dev,uio,flag)18737590Smarc ptswrite(dev, uio, flag)
1885396Sroot dev_t dev;
1897823Sroot struct uio *uio;
19052944Storek int flag;
1914484Swnj {
19235811Smarc register struct tty *tp;
1932281Stoy
19435811Smarc tp = &pt_tty[minor(dev)];
1958521Sroot if (tp->t_oproc == 0)
1968521Sroot return (EIO);
19737590Smarc return ((*linesw[tp->t_line].l_write)(tp, uio, flag));
1982281Stoy }
1992281Stoy
2005408Swnj /*
2015408Swnj * Start output on pseudo-tty.
2025408Swnj * Wake up process selecting or sleeping for input from controlling tty.
2035408Swnj */
20452411Storek void
ptsstart(tp)2052281Stoy ptsstart(tp)
2064484Swnj struct tty *tp;
2074484Swnj {
2085574Swnj register struct pt_ioctl *pti = &pt_ioctl[minor(tp->t_dev)];
2094484Swnj
2105408Swnj if (tp->t_state & TS_TTSTOP)
2112281Stoy return;
2125574Swnj if (pti->pt_flags & PF_STOPPED) {
2135574Swnj pti->pt_flags &= ~PF_STOPPED;
2145574Swnj pti->pt_send = TIOCPKT_START;
2155574Swnj }
21618651Sbloom ptcwakeup(tp, FREAD);
2175430Swnj }
2185430Swnj
21918651Sbloom ptcwakeup(tp, flag)
2205430Swnj struct tty *tp;
22152411Storek int flag;
2225430Swnj {
2235430Swnj struct pt_ioctl *pti = &pt_ioctl[minor(tp->t_dev)];
2245430Swnj
22518651Sbloom if (flag & FREAD) {
22652523Smckusick selwakeup(&pti->pt_selr);
22718651Sbloom wakeup((caddr_t)&tp->t_outq.c_cf);
2284484Swnj }
22918651Sbloom if (flag & FWRITE) {
23052523Smckusick selwakeup(&pti->pt_selw);
23118651Sbloom wakeup((caddr_t)&tp->t_rawq.c_cf);
23218651Sbloom }
2332281Stoy }
2342281Stoy
2352281Stoy /*ARGSUSED*/
23648442Skarels #ifdef __STDC__
ptcopen(dev_t dev,int flag,int devtype,struct proc * p)23748442Skarels ptcopen(dev_t dev, int flag, int devtype, struct proc *p)
23848442Skarels #else
23948442Skarels ptcopen(dev, flag, devtype, p)
2404484Swnj dev_t dev;
24148442Skarels int flag, devtype;
24248442Skarels struct proc *p;
24348442Skarels #endif
2444484Swnj {
2452281Stoy register struct tty *tp;
2465427Swnj struct pt_ioctl *pti;
2472281Stoy
24859119Storek if (minor(dev) >= npty)
2498563Sroot return (ENXIO);
2502281Stoy tp = &pt_tty[minor(dev)];
2518563Sroot if (tp->t_oproc)
2528563Sroot return (EIO);
2534484Swnj tp->t_oproc = ptsstart;
25452411Storek #ifdef sun4c
25552411Storek tp->t_stop = ptsstop;
25652411Storek #endif
25725390Skarels (void)(*linesw[tp->t_line].l_modem)(tp, 1);
25842895Smarc tp->t_lflag &= ~EXTPROC;
2595427Swnj pti = &pt_ioctl[minor(dev)];
2605427Swnj pti->pt_flags = 0;
2615427Swnj pti->pt_send = 0;
26218651Sbloom pti->pt_ucntl = 0;
2638563Sroot return (0);
2642281Stoy }
2652281Stoy
ptcclose(dev)2662281Stoy ptcclose(dev)
2674484Swnj dev_t dev;
2684484Swnj {
2692281Stoy register struct tty *tp;
2702281Stoy
2712281Stoy tp = &pt_tty[minor(dev)];
27225390Skarels (void)(*linesw[tp->t_line].l_modem)(tp, 0);
27332326Skarels tp->t_state &= ~TS_CARR_ON;
2744484Swnj tp->t_oproc = 0; /* mark closed */
27539497Smarc tp->t_session = 0;
27653645Smarc return (0);
2772281Stoy }
2782281Stoy
ptcread(dev,uio,flag)27937590Smarc ptcread(dev, uio, flag)
2805427Swnj dev_t dev;
2817823Sroot struct uio *uio;
28252411Storek int flag;
2834484Swnj {
2848521Sroot register struct tty *tp = &pt_tty[minor(dev)];
28518651Sbloom struct pt_ioctl *pti = &pt_ioctl[minor(dev)];
28616788Ssam char buf[BUFSIZ];
28716788Ssam int error = 0, cc;
2882281Stoy
28918651Sbloom /*
29018651Sbloom * We want to block until the slave
29118651Sbloom * is open, and there's something to read;
29218651Sbloom * but if we lost the slave or we're NBIO,
29318651Sbloom * then return the appropriate error instead.
29418651Sbloom */
29518651Sbloom for (;;) {
29618651Sbloom if (tp->t_state&TS_ISOPEN) {
29718651Sbloom if (pti->pt_flags&PF_PKT && pti->pt_send) {
29826358Skarels error = ureadc((int)pti->pt_send, uio);
29918651Sbloom if (error)
30018651Sbloom return (error);
30142895Smarc if (pti->pt_send & TIOCPKT_IOCTL) {
30255060Spendry cc = min(uio->uio_resid,
30342895Smarc sizeof(tp->t_termios));
30442895Smarc uiomove(&tp->t_termios, cc, uio);
30542895Smarc }
30618651Sbloom pti->pt_send = 0;
30718651Sbloom return (0);
30818651Sbloom }
30918651Sbloom if (pti->pt_flags&PF_UCNTL && pti->pt_ucntl) {
31026358Skarels error = ureadc((int)pti->pt_ucntl, uio);
31118651Sbloom if (error)
31218651Sbloom return (error);
31318651Sbloom pti->pt_ucntl = 0;
31418651Sbloom return (0);
31518651Sbloom }
31618651Sbloom if (tp->t_outq.c_cc && (tp->t_state&TS_TTSTOP) == 0)
31718651Sbloom break;
3185427Swnj }
31916788Ssam if ((tp->t_state&TS_CARR_ON) == 0)
32037590Smarc return (0); /* EOF */
32137728Smckusick if (flag & IO_NDELAY)
3228521Sroot return (EWOULDBLOCK);
32340714Skarels if (error = tsleep((caddr_t)&tp->t_outq.c_cf, TTIPRI | PCATCH,
32440714Skarels ttyin, 0))
32540714Skarels return (error);
3265411Swnj }
32735811Smarc if (pti->pt_flags & (PF_PKT|PF_UCNTL))
32818651Sbloom error = ureadc(0, uio);
32916788Ssam while (uio->uio_resid > 0 && error == 0) {
33055060Spendry cc = q_to_b(&tp->t_outq, buf, min(uio->uio_resid, BUFSIZ));
33116788Ssam if (cc <= 0)
3327823Sroot break;
33337728Smckusick error = uiomove(buf, cc, uio);
33416788Ssam }
33535811Smarc if (tp->t_outq.c_cc <= tp->t_lowat) {
33635811Smarc if (tp->t_state&TS_ASLEEP) {
33735811Smarc tp->t_state &= ~TS_ASLEEP;
33835811Smarc wakeup((caddr_t)&tp->t_outq);
33935811Smarc }
34052523Smckusick selwakeup(&tp->t_wsel);
34135811Smarc }
3428521Sroot return (error);
3432281Stoy }
3442281Stoy
34552411Storek void
ptsstop(tp,flush)3465427Swnj ptsstop(tp, flush)
3475427Swnj register struct tty *tp;
3485427Swnj int flush;
3495427Swnj {
3505427Swnj struct pt_ioctl *pti = &pt_ioctl[minor(tp->t_dev)];
35118651Sbloom int flag;
3525427Swnj
3535574Swnj /* note: FLUSHREAD and FLUSHWRITE already ok */
3545574Swnj if (flush == 0) {
3555574Swnj flush = TIOCPKT_STOP;
3565574Swnj pti->pt_flags |= PF_STOPPED;
35718651Sbloom } else
3585574Swnj pti->pt_flags &= ~PF_STOPPED;
3596119Swnj pti->pt_send |= flush;
36018651Sbloom /* change of perspective */
36118651Sbloom flag = 0;
36218651Sbloom if (flush & FREAD)
36318651Sbloom flag |= FWRITE;
36418651Sbloom if (flush & FWRITE)
36518651Sbloom flag |= FREAD;
36623633Sbloom ptcwakeup(tp, flag);
3675427Swnj }
3685427Swnj
ptcselect(dev,rw,p)36948442Skarels ptcselect(dev, rw, p)
3704484Swnj dev_t dev;
3715408Swnj int rw;
37248442Skarels struct proc *p;
3734484Swnj {
3744484Swnj register struct tty *tp = &pt_tty[minor(dev)];
3755894Swnj struct pt_ioctl *pti = &pt_ioctl[minor(dev)];
3765430Swnj int s;
3774484Swnj
37818651Sbloom if ((tp->t_state&TS_CARR_ON) == 0)
3794484Swnj return (1);
3805408Swnj switch (rw) {
3815408Swnj
3825408Swnj case FREAD:
38325945Skarels /*
38425945Skarels * Need to block timeouts (ttrstart).
38525945Skarels */
38625945Skarels s = spltty();
38718651Sbloom if ((tp->t_state&TS_ISOPEN) &&
38825436Skarels tp->t_outq.c_cc && (tp->t_state&TS_TTSTOP) == 0) {
38925436Skarels splx(s);
39025436Skarels return (1);
39125436Skarels }
39225945Skarels splx(s);
39325436Skarels /* FALLTHROUGH */
39425436Skarels
39525436Skarels case 0: /* exceptional */
39625436Skarels if ((tp->t_state&TS_ISOPEN) &&
39718651Sbloom (pti->pt_flags&PF_PKT && pti->pt_send ||
39825945Skarels pti->pt_flags&PF_UCNTL && pti->pt_ucntl))
3995408Swnj return (1);
40052523Smckusick selrecord(p, &pti->pt_selr);
4015430Swnj break;
4025408Swnj
40325436Skarels
4045408Swnj case FWRITE:
40525945Skarels if (tp->t_state&TS_ISOPEN) {
40625945Skarels if (pti->pt_flags & PF_REMOTE) {
40725945Skarels if (tp->t_canq.c_cc == 0)
40825945Skarels return (1);
40925945Skarels } else {
41025945Skarels if (tp->t_rawq.c_cc + tp->t_canq.c_cc < TTYHOG-2)
41125945Skarels return (1);
41235811Smarc if (tp->t_canq.c_cc == 0 && (tp->t_iflag&ICANON))
41325945Skarels return (1);
41425945Skarels }
4155430Swnj }
41652523Smckusick selrecord(p, &pti->pt_selw);
4175430Swnj break;
41825436Skarels
4195408Swnj }
4205430Swnj return (0);
4215396Sroot }
4224484Swnj
ptcwrite(dev,uio,flag)42337590Smarc ptcwrite(dev, uio, flag)
4245408Swnj dev_t dev;
42518651Sbloom register struct uio *uio;
42652944Storek int flag;
4274484Swnj {
4288521Sroot register struct tty *tp = &pt_tty[minor(dev)];
42945570Skarels register u_char *cp;
43018651Sbloom register int cc = 0;
43145570Skarels u_char locbuf[BUFSIZ];
4325408Swnj int cnt = 0;
4335894Swnj struct pt_ioctl *pti = &pt_ioctl[minor(dev)];
4348521Sroot int error = 0;
4352281Stoy
43618651Sbloom again:
43718651Sbloom if ((tp->t_state&TS_ISOPEN) == 0)
43818651Sbloom goto block;
43923633Sbloom if (pti->pt_flags & PF_REMOTE) {
44018651Sbloom if (tp->t_canq.c_cc)
44118651Sbloom goto block;
44245570Skarels while (uio->uio_resid > 0 && tp->t_canq.c_cc < TTYHOG - 1) {
44323633Sbloom if (cc == 0) {
44445570Skarels cc = min(uio->uio_resid, BUFSIZ);
44545570Skarels cc = min(cc, TTYHOG - 1 - tp->t_canq.c_cc);
44623633Sbloom cp = locbuf;
44745570Skarels error = uiomove((caddr_t)cp, cc, uio);
44823633Sbloom if (error)
44923633Sbloom return (error);
45023633Sbloom /* check again for safety */
45123633Sbloom if ((tp->t_state&TS_ISOPEN) == 0)
45223633Sbloom return (EIO);
45323633Sbloom }
45423633Sbloom if (cc)
45545570Skarels (void) b_to_q((char *)cp, cc, &tp->t_canq);
45623633Sbloom cc = 0;
4577823Sroot }
45818651Sbloom (void) putc(0, &tp->t_canq);
45918651Sbloom ttwakeup(tp);
46018651Sbloom wakeup((caddr_t)&tp->t_canq);
46118651Sbloom return (0);
46218651Sbloom }
46345570Skarels while (uio->uio_resid > 0) {
46418651Sbloom if (cc == 0) {
46545570Skarels cc = min(uio->uio_resid, BUFSIZ);
46618651Sbloom cp = locbuf;
46745570Skarels error = uiomove((caddr_t)cp, cc, uio);
46818651Sbloom if (error)
46918651Sbloom return (error);
47018651Sbloom /* check again for safety */
47118651Sbloom if ((tp->t_state&TS_ISOPEN) == 0)
47218651Sbloom return (EIO);
4735894Swnj }
47423633Sbloom while (cc > 0) {
47523633Sbloom if ((tp->t_rawq.c_cc + tp->t_canq.c_cc) >= TTYHOG - 2 &&
47635811Smarc (tp->t_canq.c_cc > 0 || !(tp->t_iflag&ICANON))) {
47723633Sbloom wakeup((caddr_t)&tp->t_rawq);
47823633Sbloom goto block;
47923633Sbloom }
48045570Skarels (*linesw[tp->t_line].l_rint)(*cp++, tp);
4815408Swnj cnt++;
48223633Sbloom cc--;
4832281Stoy }
48418651Sbloom cc = 0;
48518651Sbloom }
48618651Sbloom return (0);
48718651Sbloom block:
48818651Sbloom /*
48923633Sbloom * Come here to wait for slave to open, for space
49023633Sbloom * in outq, or space in rawq.
49118651Sbloom */
49218651Sbloom if ((tp->t_state&TS_CARR_ON) == 0)
49318651Sbloom return (EIO);
49445570Skarels if (flag & IO_NDELAY) {
49545570Skarels /* adjust for data copied in but not written */
49618651Sbloom uio->uio_resid += cc;
49723633Sbloom if (cnt == 0)
49823633Sbloom return (EWOULDBLOCK);
49918651Sbloom return (0);
50018651Sbloom }
50140714Skarels if (error = tsleep((caddr_t)&tp->t_rawq.c_cf, TTOPRI | PCATCH,
50245570Skarels ttyout, 0)) {
50345570Skarels /* adjust for data copied in but not written */
50445570Skarels uio->uio_resid += cc;
50540714Skarels return (error);
50645570Skarels }
50718651Sbloom goto again;
5082281Stoy }
5092281Stoy
5102281Stoy /*ARGSUSED*/
ptyioctl(dev,cmd,data,flag,p)51152411Storek ptyioctl(dev, cmd, data, flag, p)
51252411Storek dev_t dev;
51368171Scgd u_long cmd;
5147626Ssam caddr_t data;
51552411Storek int flag;
51652411Storek struct proc *p;
5174484Swnj {
5186119Swnj register struct tty *tp = &pt_tty[minor(dev)];
5196119Swnj register struct pt_ioctl *pti = &pt_ioctl[minor(dev)];
52035811Smarc register u_char *cc = tp->t_cc;
52118651Sbloom int stop, error;
5222281Stoy
52325390Skarels /*
52425390Skarels * IF CONTROLLER STTY THEN MUST FLUSH TO PREVENT A HANG.
52525390Skarels * ttywflush(tp) will hang if there are characters in the outq.
52625390Skarels */
52742895Smarc if (cmd == TIOCEXT) {
52842895Smarc /*
52942895Smarc * When the EXTPROC bit is being toggled, we need
53042895Smarc * to send an TIOCPKT_IOCTL if the packet driver
53142895Smarc * is turned on.
53242895Smarc */
53342895Smarc if (*(int *)data) {
53442895Smarc if (pti->pt_flags & PF_PKT) {
53542895Smarc pti->pt_send |= TIOCPKT_IOCTL;
53663689Smckusick ptcwakeup(tp, FREAD);
53742895Smarc }
53842895Smarc tp->t_lflag |= EXTPROC;
53942895Smarc } else {
540*68389Sdab if ((tp->t_lflag & EXTPROC) &&
54142895Smarc (pti->pt_flags & PF_PKT)) {
54242895Smarc pti->pt_send |= TIOCPKT_IOCTL;
54363689Smckusick ptcwakeup(tp, FREAD);
54442895Smarc }
54542895Smarc tp->t_lflag &= ~EXTPROC;
54642895Smarc }
54742895Smarc return(0);
54842895Smarc } else
54935811Smarc if (cdevsw[major(dev)].d_open == ptcopen)
5507626Ssam switch (cmd) {
5517626Ssam
55244313Smarc case TIOCGPGRP:
55344313Smarc /*
55444313Smarc * We aviod calling ttioctl on the controller since,
55544313Smarc * in that case, tp must be the controlling terminal.
55644313Smarc */
55744313Smarc *(int *)data = tp->t_pgrp ? tp->t_pgrp->pg_id : 0;
55844313Smarc return (0);
55944313Smarc
5607626Ssam case TIOCPKT:
56118651Sbloom if (*(int *)data) {
56218651Sbloom if (pti->pt_flags & PF_UCNTL)
56318651Sbloom return (EINVAL);
5645427Swnj pti->pt_flags |= PF_PKT;
56518651Sbloom } else
5665427Swnj pti->pt_flags &= ~PF_PKT;
5678563Sroot return (0);
5687626Ssam
56918651Sbloom case TIOCUCNTL:
57018651Sbloom if (*(int *)data) {
57118651Sbloom if (pti->pt_flags & PF_PKT)
57218651Sbloom return (EINVAL);
57318651Sbloom pti->pt_flags |= PF_UCNTL;
57418651Sbloom } else
57518651Sbloom pti->pt_flags &= ~PF_UCNTL;
57618651Sbloom return (0);
57718651Sbloom
5787626Ssam case TIOCREMOTE:
5797626Ssam if (*(int *)data)
5805894Swnj pti->pt_flags |= PF_REMOTE;
5815894Swnj else
5825894Swnj pti->pt_flags &= ~PF_REMOTE;
58312753Ssam ttyflush(tp, FREAD|FWRITE);
5848563Sroot return (0);
5857626Ssam
58652411Storek #ifdef COMPAT_43
58735811Smarc case TIOCSETP:
58825390Skarels case TIOCSETN:
58952411Storek #endif
59025390Skarels case TIOCSETD:
59135811Smarc case TIOCSETA:
59235811Smarc case TIOCSETAW:
59335811Smarc case TIOCSETAF:
59452411Storek ndflush(&tp->t_outq, tp->t_outq.c_cc);
5957626Ssam break;
59642895Smarc
59742895Smarc case TIOCSIG:
59842895Smarc if (*(unsigned int *)data >= NSIG)
59942895Smarc return(EINVAL);
60042895Smarc if ((tp->t_lflag&NOFLSH) == 0)
60142895Smarc ttyflush(tp, FREAD|FWRITE);
60245227Sborman pgsignal(tp->t_pgrp, *(unsigned int *)data, 1);
60345227Sborman if ((*(unsigned int *)data == SIGINFO) &&
60445227Sborman ((tp->t_lflag&NOKERNINFO) == 0))
60545227Sborman ttyinfo(tp);
60642895Smarc return(0);
6075411Swnj }
60852411Storek error = (*linesw[tp->t_line].l_ioctl)(tp, cmd, data, flag, p);
60935811Smarc if (error < 0)
61035811Smarc error = ttioctl(tp, cmd, data, flag);
61118651Sbloom if (error < 0) {
61218651Sbloom if (pti->pt_flags & PF_UCNTL &&
61328285Skarels (cmd & ~0xff) == UIOCCMD(0)) {
61418651Sbloom if (cmd & 0xff) {
61518651Sbloom pti->pt_ucntl = (u_char)cmd;
61618651Sbloom ptcwakeup(tp, FREAD);
61718651Sbloom }
61818651Sbloom return (0);
61918651Sbloom }
6208563Sroot error = ENOTTY;
62118651Sbloom }
62242895Smarc /*
62342895Smarc * If external processing and packet mode send ioctl packet.
62442895Smarc */
62542895Smarc if ((tp->t_lflag&EXTPROC) && (pti->pt_flags & PF_PKT)) {
62642895Smarc switch(cmd) {
62742895Smarc case TIOCSETA:
62842895Smarc case TIOCSETAW:
62942895Smarc case TIOCSETAF:
63052411Storek #ifdef COMPAT_43
63142895Smarc case TIOCSETP:
63242895Smarc case TIOCSETN:
63352411Storek #endif
63452411Storek #if defined(COMPAT_43) || defined(COMPAT_SUNOS)
63542895Smarc case TIOCSETC:
63642895Smarc case TIOCSLTC:
63742895Smarc case TIOCLBIS:
63842895Smarc case TIOCLBIC:
63942895Smarc case TIOCLSET:
64042895Smarc #endif
64142895Smarc pti->pt_send |= TIOCPKT_IOCTL;
64263689Smckusick ptcwakeup(tp, FREAD);
64342895Smarc default:
64442895Smarc break;
64542895Smarc }
64642895Smarc }
64735811Smarc stop = (tp->t_iflag & IXON) && CCEQ(cc[VSTOP], CTRL('s'))
64835811Smarc && CCEQ(cc[VSTART], CTRL('q'));
6496119Swnj if (pti->pt_flags & PF_NOSTOP) {
6506119Swnj if (stop) {
65125478Skarels pti->pt_send &= ~TIOCPKT_NOSTOP;
6526119Swnj pti->pt_send |= TIOCPKT_DOSTOP;
6536119Swnj pti->pt_flags &= ~PF_NOSTOP;
65418651Sbloom ptcwakeup(tp, FREAD);
6556119Swnj }
6566119Swnj } else {
65718651Sbloom if (!stop) {
6586119Swnj pti->pt_send &= ~TIOCPKT_DOSTOP;
6596119Swnj pti->pt_send |= TIOCPKT_NOSTOP;
6606119Swnj pti->pt_flags |= PF_NOSTOP;
66118651Sbloom ptcwakeup(tp, FREAD);
6626119Swnj }
6636119Swnj }
6648563Sroot return (error);
6652281Stoy }
666