summaryrefslogtreecommitdiff
path: root/sys/arch/sparc64/dev/z8530kbd.c
diff options
context:
space:
mode:
authorJason Wright <jason@cvs.openbsd.org>2002-01-16 21:33:00 +0000
committerJason Wright <jason@cvs.openbsd.org>2002-01-16 21:33:00 +0000
commit235fa20ab8e9823c8c30423e5953647b82f6a816 (patch)
tree4c7be75d216bec7fa076011b3869a2d269d8f94d /sys/arch/sparc64/dev/z8530kbd.c
parent37a7bf516bc2dddbde1d5020b8fad58b737fb2b3 (diff)
More removal of tty stuff
add support for updating LED status add ScrollLock keysym [Committed from the console of a u1 with wsdisplay at cgsix and wskbd at zskbd =]
Diffstat (limited to 'sys/arch/sparc64/dev/z8530kbd.c')
-rw-r--r--sys/arch/sparc64/dev/z8530kbd.c348
1 files changed, 60 insertions, 288 deletions
diff --git a/sys/arch/sparc64/dev/z8530kbd.c b/sys/arch/sparc64/dev/z8530kbd.c
index deb2af2b5c8..62ececffd13 100644
--- a/sys/arch/sparc64/dev/z8530kbd.c
+++ b/sys/arch/sparc64/dev/z8530kbd.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: z8530kbd.c,v 1.4 2002/01/16 18:04:42 jason Exp $ */
+/* $OpenBSD: z8530kbd.c,v 1.5 2002/01/16 21:32:59 jason Exp $ */
/* $NetBSD: z8530tty.c,v 1.77 2001/05/30 15:24:24 lukem Exp $ */
/*-
@@ -147,7 +147,6 @@ u_int zskbd_rbuf_lowat = (ZSKBD_RING_SIZE * 3) / 4;
struct zskbd_softc {
struct device zst_dev; /* required first: base device */
- struct tty *zst_tty;
struct zs_chanstate *zst_cs;
struct timeout zst_diag_ch;
@@ -177,6 +176,9 @@ struct zskbd_softc {
u_int zst_tbc, /* transmit byte count */
zst_heldtbc; /* held tbc while xmission stopped */
+ u_char zst_tbuf[ZSKBD_RING_SIZE];
+ u_char *zst_tbeg, *zst_tend, *zst_tbp;
+
/* Flags to communicate with zskbd_softint() */
volatile u_char zst_rx_flags, /* receiver blocked */
#define RX_TTY_BLOCKED 0x01
@@ -216,10 +218,7 @@ struct cfattach zskbd_ca = {
struct zsops zsops_kbd;
-static void zsstart __P((struct tty *));
-static int zsparam __P((struct tty *, struct termios *));
static void zs_modem __P((struct zskbd_softc *, int));
-static int zshwiflow __P((struct tty *, int));
static void zs_hwiflow __P((struct zskbd_softc *));
static void zs_maskintr __P((struct zskbd_softc *));
@@ -244,6 +243,9 @@ int zskbd_ioctl __P((void *, u_long, caddr_t, int, struct proc *));
void zskbd_cngetc __P((void *, u_int *, int *));
void zskbd_cnpollc __P((void *, int));
+void zsstart_tx __P((struct zskbd_softc *));
+int zsenqueue_tx __P((struct zskbd_softc *, u_char *, int));
+
struct wskbd_accessops zskbd_accessops = {
zskbd_enable,
zskbd_set_leds,
@@ -258,8 +260,8 @@ struct wskbd_consops zskbd_consops = {
#define KC(n) KS_KEYCODE(n)
const keysym_t zskbd_keydesc_us[] = {
KC(0x01), KS_Cmd,
- KC(0x02), KS_Cmd_BrightnessDown,
- KC(0x04), KS_Cmd_BrightnessUp,
+ KC(0x02), KS_Cmd_BrightnessDown,
+ KC(0x04), KS_Cmd_BrightnessUp,
KC(0x05), KS_f1,
KC(0x06), KS_f2,
KC(0x07), KS_f10,
@@ -278,6 +280,7 @@ const keysym_t zskbd_keydesc_us[] = {
KC(0x15), KS_Pause,
KC(0x16), KS_Print_Screen,
KC(0x18), KS_Left,
+ KC(0x19), KS_Hold_Screen,
KC(0x1b), KS_Down,
KC(0x1c), KS_Right,
KC(0x1d), KS_Escape,
@@ -432,12 +435,14 @@ zskbd_attach(parent, self, aux)
struct zsc_attach_args *args = aux;
struct wskbddev_attach_args a;
struct zs_chanstate *cs;
- struct tty *tp;
int channel, s, tty_unit, console = 0;
dev_t dev;
timeout_set(&zst->zst_diag_ch, zskbd_diag, zst);
+ zst->zst_tbp = zst->zst_tba = zst->zst_tbeg = zst->zst_tbuf;
+ zst->zst_tend = zst->zst_tbeg + ZSKBD_RING_SIZE;
+
tty_unit = zst->zst_dev.dv_unit;
channel = args->channel;
cs = zsc->zsc_cs[channel];
@@ -467,14 +472,6 @@ zskbd_attach(parent, self, aux)
console = 1;
}
- tp = ttymalloc();
- tp->t_dev = dev;
- tp->t_oproc = zsstart;
- tp->t_param = zsparam;
- tp->t_hwiflow = zshwiflow;
- tty_attach(tp);
-
- zst->zst_tty = tp;
zst->zst_rbuf = malloc(zskbd_rbuf_size << 1, M_DEVBUF, M_WAITOK);
zst->zst_ebuf = zst->zst_rbuf + (zskbd_rbuf_size << 1);
/* Disable the high water mark. */
@@ -640,7 +637,7 @@ zskbd_init(zst)
*/
if (zst->zst_tx_stopped) {
zst->zst_tx_stopped = 0;
- zsstart(zst->zst_tty);
+ zsstart_tx(zst);
}
zskbd_softint(cs);
@@ -769,245 +766,62 @@ zskbd_putc(zst, c)
splx(s);
}
-/*
- * Start or restart transmission.
- */
-static void
-zsstart(tp)
- struct tty *tp;
+int
+zsenqueue_tx(zst, str, len)
+ struct zskbd_softc *zst;
+ u_char *str;
+ int len;
+{
+ int s, i;
+
+ s = splzs();
+ if (zst->zst_tbc + len > ZSKBD_RING_SIZE)
+ return (-1);
+ zst->zst_tbc += len;
+ for (i = 0; i < len; i++) {
+ *zst->zst_tbp = str[i];
+ if (++zst->zst_tbp == zst->zst_tend)
+ zst->zst_tbp = zst->zst_tbeg;
+ }
+ splx(s);
+ zsstart_tx(zst);
+ return (0);
+}
+
+void
+zsstart_tx(zst)
+ struct zskbd_softc *zst;
{
- struct zskbd_softc *zst = zskbd_device_lookup(&zskbd_cd, ZSKBDUNIT(tp->t_dev));
struct zs_chanstate *cs = zst->zst_cs;
- int s;
+ int s, s1;
s = spltty();
- if (ISSET(tp->t_state, TS_BUSY | TS_TIMEOUT | TS_TTSTOP))
- goto out;
+
if (zst->zst_tx_stopped)
goto out;
+ if (zst->zst_tbc == 0)
+ goto out;
- if (tp->t_outq.c_cc <= tp->t_lowat) {
- if (ISSET(tp->t_state, TS_ASLEEP)) {
- CLR(tp->t_state, TS_ASLEEP);
- wakeup((caddr_t)&tp->t_outq);
- }
- selwakeup(&tp->t_wsel);
- if (tp->t_outq.c_cc == 0)
- goto out;
- }
-
- /* Grab the first contiguous region of buffer space. */
- {
- u_char *tba;
- int tbc;
-
- tba = tp->t_outq.c_cf;
- tbc = ndqb(&tp->t_outq, 0);
-
- (void) splzs();
-
- zst->zst_tba = tba;
- zst->zst_tbc = tbc;
- }
+ s1 = splzs();
- SET(tp->t_state, TS_BUSY);
zst->zst_tx_busy = 1;
- /* Enable transmit completion interrupts if necessary. */
if (!ISSET(cs->cs_preg[1], ZSWR1_TIE)) {
SET(cs->cs_preg[1], ZSWR1_TIE);
cs->cs_creg[1] = cs->cs_preg[1];
zs_write_reg(cs, 1, cs->cs_creg[1]);
}
- /* Output the first character of the contiguous buffer. */
- {
- zs_write_data(cs, *zst->zst_tba);
- zst->zst_tbc--;
- zst->zst_tba++;
- }
-out:
- splx(s);
- return;
-}
-
-/*
- * Set ZS tty parameters from termios.
- * XXX - Should just copy the whole termios after
- * making sure all the changes could be done.
- */
-static int
-zsparam(tp, t)
- struct tty *tp;
- struct termios *t;
-{
- struct zskbd_softc *zst = zskbd_device_lookup(&zskbd_cd, ZSKBDUNIT(tp->t_dev));
- struct zs_chanstate *cs = zst->zst_cs;
- int ospeed, cflag;
- u_char tmp3, tmp4, tmp5;
- int s, error;
+ zs_write_data(cs, *zst->zst_tba);
- ospeed = t->c_ospeed;
- cflag = t->c_cflag;
+ zst->zst_tbc--;
+ if (++zst->zst_tba == zst->zst_tend)
+ zst->zst_tba = zst->zst_tbeg;
- /* Check requested parameters. */
- if (ospeed < 0)
- return (EINVAL);
- if (t->c_ispeed && t->c_ispeed != ospeed)
- return (EINVAL);
-
- /*
- * For the console, always force CLOCAL and !HUPCL, so that the port
- * is always active.
- */
- if (ISSET(zst->zst_swflags, TIOCFLAG_SOFTCAR) ||
- ISSET(zst->zst_hwflags, ZS_HWFLAG_CONSOLE)) {
- SET(cflag, CLOCAL);
- CLR(cflag, HUPCL);
- }
-
- /*
- * Only whack the UART when params change.
- * Some callers need to clear tp->t_ospeed
- * to make sure initialization gets done.
- */
- if (tp->t_ospeed == ospeed &&
- tp->t_cflag == cflag)
- return (0);
-
- /*
- * Call MD functions to deal with changed
- * clock modes or H/W flow control modes.
- * The BRG divisor is set now. (reg 12,13)
- */
- error = zs_set_speed(cs, ospeed);
- if (error)
- return (error);
- error = zs_set_modes(cs, cflag);
- if (error)
- return (error);
-
- /*
- * Block interrupts so that state will not
- * be altered until we are done setting it up.
- *
- * Initial values in cs_preg are set before
- * our attach routine is called. The master
- * interrupt enable is handled by zsc.c
- *
- */
- s = splzs();
-
- /*
- * Recalculate which status ints to enable.
- */
- zs_maskintr(zst);
-
- /* Recompute character size bits. */
- tmp3 = cs->cs_preg[3];
- tmp5 = cs->cs_preg[5];
- CLR(tmp3, ZSWR3_RXSIZE);
- CLR(tmp5, ZSWR5_TXSIZE);
- switch (ISSET(cflag, CSIZE)) {
- case CS5:
- SET(tmp3, ZSWR3_RX_5);
- SET(tmp5, ZSWR5_TX_5);
- break;
- case CS6:
- SET(tmp3, ZSWR3_RX_6);
- SET(tmp5, ZSWR5_TX_6);
- break;
- case CS7:
- SET(tmp3, ZSWR3_RX_7);
- SET(tmp5, ZSWR5_TX_7);
- break;
- case CS8:
- SET(tmp3, ZSWR3_RX_8);
- SET(tmp5, ZSWR5_TX_8);
- break;
- }
- cs->cs_preg[3] = tmp3;
- cs->cs_preg[5] = tmp5;
-
- /*
- * Recompute the stop bits and parity bits. Note that
- * zs_set_speed() may have set clock selection bits etc.
- * in wr4, so those must preserved.
- */
- tmp4 = cs->cs_preg[4];
- CLR(tmp4, ZSWR4_SBMASK | ZSWR4_PARMASK);
- if (ISSET(cflag, CSTOPB))
- SET(tmp4, ZSWR4_TWOSB);
- else
- SET(tmp4, ZSWR4_ONESB);
- if (!ISSET(cflag, PARODD))
- SET(tmp4, ZSWR4_EVENP);
- if (ISSET(cflag, PARENB))
- SET(tmp4, ZSWR4_PARENB);
- cs->cs_preg[4] = tmp4;
-
- /* And copy to tty. */
- tp->t_ispeed = 0;
- tp->t_ospeed = ospeed;
- tp->t_cflag = cflag;
-
- /*
- * If nothing is being transmitted, set up new current values,
- * else mark them as pending.
- */
- if (!cs->cs_heldchange) {
- if (zst->zst_tx_busy) {
- zst->zst_heldtbc = zst->zst_tbc;
- zst->zst_tbc = 0;
- cs->cs_heldchange = 1;
- } else
- zs_loadchannelregs(cs);
- }
-
- /*
- * If hardware flow control is disabled, turn off the buffer water
- * marks and unblock any soft flow control state. Otherwise, enable
- * the water marks.
- */
- if (!ISSET(cflag, CHWFLOW)) {
- zst->zst_r_hiwat = 0;
- zst->zst_r_lowat = 0;
- if (ISSET(zst->zst_rx_flags, RX_TTY_OVERFLOWED)) {
- CLR(zst->zst_rx_flags, RX_TTY_OVERFLOWED);
- zst->zst_rx_ready = 1;
- cs->cs_softreq = 1;
- }
- if (ISSET(zst->zst_rx_flags, RX_TTY_BLOCKED|RX_IBUF_BLOCKED)) {
- CLR(zst->zst_rx_flags, RX_TTY_BLOCKED|RX_IBUF_BLOCKED);
- zs_hwiflow(zst);
- }
- } else {
- zst->zst_r_hiwat = zskbd_rbuf_hiwat;
- zst->zst_r_lowat = zskbd_rbuf_lowat;
- }
-
- /*
- * Force a recheck of the hardware carrier and flow control status,
- * since we may have changed which bits we're looking at.
- */
- zskbd_stint(cs, 1);
+ splx(s1);
+out:
splx(s);
-
- /*
- * If hardware flow control is disabled, unblock any hard flow control
- * state.
- */
- if (!ISSET(cflag, CHWFLOW)) {
- if (zst->zst_tx_stopped) {
- zst->zst_tx_stopped = 0;
- zsstart(tp);
- }
- }
-
- zskbd_softint(cs);
-
- return (0);
}
/*
@@ -1068,45 +882,6 @@ zs_modem(zst, onoff)
}
/*
- * Try to block or unblock input using hardware flow-control.
- * This is called by kern/tty.c if MDMBUF|CRTSCTS is set, and
- * if this function returns non-zero, the TS_TBLOCK flag will
- * be set or cleared according to the "block" arg passed.
- */
-int
-zshwiflow(tp, block)
- struct tty *tp;
- int block;
-{
- struct zskbd_softc *zst = zskbd_device_lookup(&zskbd_cd, ZSKBDUNIT(tp->t_dev));
- struct zs_chanstate *cs = zst->zst_cs;
- int s;
-
- if (cs->cs_wr5_rts == 0)
- return (0);
-
- s = splzs();
- if (block) {
- if (!ISSET(zst->zst_rx_flags, RX_TTY_BLOCKED)) {
- SET(zst->zst_rx_flags, RX_TTY_BLOCKED);
- zs_hwiflow(zst);
- }
- } else {
- if (ISSET(zst->zst_rx_flags, RX_TTY_OVERFLOWED)) {
- CLR(zst->zst_rx_flags, RX_TTY_OVERFLOWED);
- zst->zst_rx_ready = 1;
- cs->cs_softreq = 1;
- }
- if (ISSET(zst->zst_rx_flags, RX_TTY_BLOCKED)) {
- CLR(zst->zst_rx_flags, RX_TTY_BLOCKED);
- zs_hwiflow(zst);
- }
- }
- splx(s);
- return (1);
-}
-
-/*
* Internal version of zshwiflow
* called at splzs
*/
@@ -1136,7 +911,7 @@ zs_hwiflow(zst)
#define integrate
integrate void zskbd_rxsoft __P((struct zskbd_softc *));
-integrate void zskbd_txsoft __P((struct zskbd_softc *, struct tty *));
+integrate void zskbd_txsoft __P((struct zskbd_softc *));
integrate void zskbd_stsoft __P((struct zskbd_softc *));
/*
* receiver ready interrupt.
@@ -1239,7 +1014,8 @@ zskbd_txint(cs)
if (zst->zst_tbc > 0) {
zs_write_data(cs, *zst->zst_tba);
zst->zst_tbc--;
- zst->zst_tba++;
+ if (++zst->zst_tba == zst->zst_tend)
+ zst->zst_tba = zst->zst_tbeg;
} else {
/* Disable transmit completion interrupts if necessary. */
if (ISSET(cs->cs_preg[1], ZSWR1_TIE)) {
@@ -1395,17 +1171,9 @@ zskbd_rxsoft(zst)
}
integrate void
-zskbd_txsoft(zst, tp)
+zskbd_txsoft(zst)
struct zskbd_softc *zst;
- struct tty *tp;
{
-
- CLR(tp->t_state, TS_BUSY);
- if (ISSET(tp->t_state, TS_FLUSH))
- CLR(tp->t_state, TS_FLUSH);
- else
- ndflush(&tp->t_outq, (int)(zst->zst_tba - tp->t_outq.c_cf));
- (*linesw[tp->t_line].l_start)(tp);
}
integrate void
@@ -1448,7 +1216,6 @@ zskbd_softint(cs)
struct zs_chanstate *cs;
{
struct zskbd_softc *zst = cs->cs_private;
- struct tty *tp = zst->zst_tty;
int s;
s = spltty();
@@ -1465,7 +1232,7 @@ zskbd_softint(cs)
if (zst->zst_tx_done) {
zst->zst_tx_done = 0;
- zskbd_txsoft(zst, tp);
+ zskbd_txsoft(zst);
}
splx(s);
@@ -1493,6 +1260,7 @@ zskbd_set_leds(v, wled)
{
struct zskbd_softc *zst = v;
u_int8_t sled = 0;
+ u_int8_t cmd[2];
zst->zst_leds = wled;
@@ -1504,6 +1272,10 @@ zskbd_set_leds(v, wled)
sled |= SKBD_LED_SCROLLLOCK;
if (wled & WSKBD_LED_COMPOSE)
sled |= SKBD_LED_COMPOSE;
+
+ cmd[0] = SKBD_CMD_SETLED;
+ cmd[1] = sled;
+ zsenqueue_tx(zst, cmd, sizeof(cmd));
}
int