summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/arch/sparc/dev/zs.c106
-rw-r--r--sys/arch/sparc/dev/zsvar.h3
2 files changed, 90 insertions, 19 deletions
diff --git a/sys/arch/sparc/dev/zs.c b/sys/arch/sparc/dev/zs.c
index 531357ffed2..7f76d5e0a52 100644
--- a/sys/arch/sparc/dev/zs.c
+++ b/sys/arch/sparc/dev/zs.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: zs.c,v 1.20 1998/03/04 14:21:29 jason Exp $ */
+/* $OpenBSD: zs.c,v 1.21 1998/07/21 22:33:42 marc Exp $ */
/* $NetBSD: zs.c,v 1.49 1997/08/31 21:26:37 pk Exp $ */
/*
@@ -87,6 +87,14 @@
#include <machine/remote-sl.h>
#endif
+#define DEVUNIT(x) (minor(x) & 0x7f)
+#define DEVCUA(x) (minor(x) & 0x80)
+
+/* Macros to clear/set/test flags. */
+#define SET(t, f) (t) |= (f)
+#define CLR(t, f) (t) &= ~(f)
+#define ISSET(t, f) ((t) & (f))
+
#define ZSMAJOR 12 /* XXX */
#define ZS_KBD 2 /* XXX */
@@ -608,6 +616,7 @@ zsiclose(tp)
}
+
/*
* Open a zs serial port. This interface may not be used to open
* the keyboard and mouse ports. (XXX)
@@ -622,7 +631,8 @@ zsopen(dev, flags, mode, p)
register struct tty *tp;
register struct zs_chanstate *cs;
struct zs_softc *sc;
- int unit = minor(dev), zs = unit >> 1, error, s;
+ int unit = DEVUNIT(dev);
+ int zs = unit >> 1, error, s;
if (zs >= zs_cd.cd_ndevs || (sc = zs_cd.cd_devs[zs]) == NULL ||
unit == ZS_KBD || unit == ZS_MOUSE)
@@ -643,14 +653,48 @@ zsopen(dev, flags, mode, p)
}
(void) zsparam(tp, &tp->t_termios);
ttsetwater(tp);
+/* XXX start CUA mods */
+ if (DEVCUA(dev))
+ SET(tp->t_state, TS_CARR_ON);
+ else
+ CLR(tp->t_state, TS_CARR_ON);
+/* end CUA mods */
+
} else if (tp->t_state & TS_XCLUDE && p->p_ucred->cr_uid != 0) {
splx(s);
return (EBUSY);
}
- error = 0;
- for (;;) {
+
+/* XXX start CUA mods */
+ if (DEVCUA(dev)) {
+ if (ISSET(tp->t_state, TS_ISOPEN)) {
+ /* Ah, but someone already is dialed in... */
+ splx(s);
+ return EBUSY;
+ }
+ cs->cs_cua = 1; /* We go into CUA mode */
+ }
+
+
+ error = 0;
+ /* wait for carrier if necessary */
+ if (ISSET(flags, O_NONBLOCK)) {
+ if (!DEVCUA(dev) && cs->cs_cua) {
+ /* Opening TTY non-blocking... but the CUA is busy */
+ splx(s);
+ return EBUSY;
+ }
+ } else {
+ while (cs->cs_cua ||
+ (!ISSET(tp->t_cflag, CLOCAL) &&
+ !ISSET(tp->t_state, TS_CARR_ON))) {
register int rr0;
+ error = 0;
+ SET(tp->t_state, TS_WOPEN);
+
+
+ if (!DEVCUA(dev) && !cs->cs_cua) {
/* loop, turning on the device, until carrier present */
zs_modem(cs, 1);
/* May never get status intr if carrier already on. -gwr */
@@ -658,23 +702,39 @@ zsopen(dev, flags, mode, p)
ZS_DELAY();
if ((rr0 & ZSRR0_DCD) || cs->cs_softcar)
tp->t_state |= TS_CARR_ON;
- if (flags & O_NONBLOCK || tp->t_cflag & CLOCAL ||
- tp->t_state & TS_CARR_ON)
+ }
+
+ if ((tp->t_cflag & CLOCAL || tp->t_state & TS_CARR_ON) &&
+ !cs->cs_cua)
break;
- tp->t_state |= TS_WOPEN;
+
error = ttysleep(tp, (caddr_t)&tp->t_rawq, TTIPRI | PCATCH,
ttopen, 0);
+
+ if (!DEVCUA(dev) && cs->cs_cua && error == EINTR) {
+ error=0;
+ continue;
+ }
+
if (error) {
if (!(tp->t_state & TS_ISOPEN)) {
zs_modem(cs, 0);
- tp->t_state &= ~TS_WOPEN;
+ CLR(tp->t_state, TS_WOPEN);
ttwakeup(tp);
}
+/* XXX ordering of this might be important?? */
+ if (DEVCUA(dev))
+ cs->cs_cua = 0;
+ CLR(tp->t_state, TS_WOPEN);
splx(s);
return error;
}
- }
+ if (!DEVCUA(dev) && cs->cs_cua)
+ continue;
+ }
+ }
splx(s);
+/* end CUA mods */
if (error == 0)
error = linesw[tp->t_line].l_open(dev, tp);
if (error)
@@ -695,12 +755,17 @@ zsclose(dev, flags, mode, p)
register struct zs_chanstate *cs;
register struct tty *tp;
struct zs_softc *sc;
- int unit = minor(dev), s;
+ int unit = DEVUNIT(dev);
+ int s, st;
sc = zs_cd.cd_devs[unit >> 1];
cs = &sc->sc_cs[unit & 1];
tp = cs->cs_ttyp;
linesw[tp->t_line].l_close(tp, flags);
+
+/* XXX start CUA mods */
+ st = spltty();
+/* end CUA mods */
if (tp->t_cflag & HUPCL || tp->t_state & TS_WOPEN ||
(tp->t_state & TS_ISOPEN) == 0) {
zs_modem(cs, 0);
@@ -715,6 +780,11 @@ zsclose(dev, flags, mode, p)
ZS_WRITE(cs->cs_zc, 5, cs->cs_creg[5]);
splx(s);
}
+/* XXX start CUA mods */
+ CLR(tp->t_state, TS_CARR_ON | TS_BUSY | TS_FLUSH);
+ cs->cs_cua = 0;
+ splx(st);
+/* end CUA mods */
ttyclose(tp);
#ifdef KGDB
/* Reset the speed if we're doing kgdb on this port */
@@ -738,7 +808,7 @@ zsread(dev, uio, flags)
register struct zs_chanstate *cs;
register struct zs_softc *sc;
register struct tty *tp;
- int unit = minor(dev);
+ int unit = DEVUNIT(dev);
sc = zs_cd.cd_devs[unit >> 1];
cs = &sc->sc_cs[unit & 1];
@@ -757,7 +827,7 @@ zswrite(dev, uio, flags)
register struct zs_chanstate *cs;
register struct zs_softc *sc;
register struct tty *tp;
- int unit = minor(dev);
+ int unit = DEVUNIT(dev);
sc = zs_cd.cd_devs[unit >> 1];
cs = &sc->sc_cs[unit & 1];
@@ -772,7 +842,7 @@ zstty(dev)
{
register struct zs_chanstate *cs;
register struct zs_softc *sc;
- int unit = minor(dev);
+ int unit = DEVUNIT(dev);
sc = zs_cd.cd_devs[unit >> 1];
cs = &sc->sc_cs[unit & 1];
@@ -1201,7 +1271,7 @@ zsioctl(dev, cmd, data, flag, p)
int flag;
struct proc *p;
{
- int unit = minor(dev);
+ int unit = DEVUNIT(dev);
struct zs_softc *sc = zs_cd.cd_devs[unit >> 1];
register struct zs_chanstate *cs = &sc->sc_cs[unit & 1];
register struct tty *tp = cs->cs_ttyp;
@@ -1339,7 +1409,7 @@ zsstart(tp)
{
register struct zs_chanstate *cs;
register int s, nch;
- int unit = minor(tp->t_dev);
+ int unit = DEVUNIT(tp->t_dev);
struct zs_softc *sc = zs_cd.cd_devs[unit >> 1];
cs = &sc->sc_cs[unit & 1];
@@ -1400,7 +1470,7 @@ zsstop(tp, flag)
int flag;
{
register struct zs_chanstate *cs;
- register int s, unit = minor(tp->t_dev);
+ register int s, unit = DEVUNIT(tp->t_dev);
struct zs_softc *sc = zs_cd.cd_devs[unit >> 1];
cs = &sc->sc_cs[unit & 1];
@@ -1428,7 +1498,7 @@ zsparam(tp, t)
register struct tty *tp;
register struct termios *t;
{
- int unit = minor(tp->t_dev);
+ int unit = DEVUNIT(tp->t_dev);
struct zs_softc *sc = zs_cd.cd_devs[unit >> 1];
register struct zs_chanstate *cs = &sc->sc_cs[unit & 1];
register int tmp, tmp5, cflag, s;
@@ -1686,7 +1756,7 @@ zs_kgdb_init()
if (major(kgdb_dev) != ZSMAJOR)
return;
- unit = minor(kgdb_dev);
+ unit = DEVUNIT(kgdb_dev);
/*
* Unit must be 0 or 1 (zs0).
*/
diff --git a/sys/arch/sparc/dev/zsvar.h b/sys/arch/sparc/dev/zsvar.h
index 79b8c6cd2c9..a9947fc4db2 100644
--- a/sys/arch/sparc/dev/zsvar.h
+++ b/sys/arch/sparc/dev/zsvar.h
@@ -1,4 +1,4 @@
-/* $OpenBSD: zsvar.h,v 1.10 1997/08/08 08:25:44 downsj Exp $ */
+/* $OpenBSD: zsvar.h,v 1.11 1998/07/21 22:33:44 marc Exp $ */
/* $NetBSD: zsvar.h,v 1.10 1997/04/14 21:26:28 pk Exp $ */
/*
@@ -127,6 +127,7 @@ struct zs_chanstate {
char cs_consio; /* port does /dev/console I/O */
char cs_xxx; /* (spare) */
char cs_deferred_cc; /* deferred zscnputc() output */
+ u_char cs_cua; /* CUA mode flag */
int cs_speed; /* default baud rate (from ROM) */
/*