summaryrefslogtreecommitdiff
path: root/sys/arch/alpha/wscons/kbd.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/arch/alpha/wscons/kbd.c')
-rw-r--r--sys/arch/alpha/wscons/kbd.c334
1 files changed, 334 insertions, 0 deletions
diff --git a/sys/arch/alpha/wscons/kbd.c b/sys/arch/alpha/wscons/kbd.c
new file mode 100644
index 00000000000..67cd26fcee5
--- /dev/null
+++ b/sys/arch/alpha/wscons/kbd.c
@@ -0,0 +1,334 @@
+/* $NetBSD: kbd.c,v 1.1 1996/04/12 02:00:46 cgd Exp $ */
+
+/*
+ * Copyright (c) 1992, 1993
+ * The Regents of the University of California. All rights reserved.
+ *
+ * This software was developed by the Computer Systems Engineering group
+ * at Lawrence Berkeley Laboratory under DARPA contract BG 91-66 and
+ * contributed to Berkeley.
+ *
+ * All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by the University of
+ * California, Lawrence Berkeley Laboratory.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. All advertising materials mentioning features or use of this software
+ * must display the following acknowledgement:
+ * This product includes software developed by the University of
+ * California, Berkeley and its contributors.
+ * 4. Neither the name of the University nor the names of its contributors
+ * may be used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ *
+ * @(#)kbd.c 8.2 (Berkeley) 10/30/93
+ */
+
+/*
+ * Keyboard driver (/dev/kbd -- note that we do not have minor numbers
+ * [yet?]). Translates incoming bytes to ASCII or to `firm_events' and
+ * passes them up to the appropriate reader.
+ */
+
+#include <sys/param.h>
+#include <sys/conf.h>
+#include <sys/device.h>
+#include <sys/ioctl.h>
+#include <sys/kernel.h>
+#include <sys/proc.h>
+#include <sys/syslog.h>
+#include <sys/systm.h>
+#include <sys/tty.h>
+
+#include <machine/autoconf.h>
+
+#include <machine/vuid_event.h>
+#include <machine/kbio.h> /* XXX FOR KIOCSDIRECT */
+#include <machine/wsconsio.h> /* XXX for bell ioctls */
+#include <alpha/wscons/event_var.h>
+#include <alpha/wscons/wsconsvar.h>
+
+struct kbd_softc {
+ struct device *k_idev; /* the input device */
+ struct wscons_idev_spec k_ispec; /* the input device information */
+
+ int k_evmode; /* set if we should produce events */
+ struct evvar k_events; /* event queue state */
+ char *k_repeatcp; /* repeated character (string) */
+ int k_repeating; /* we've called timeout() */
+ int k_repeat_start; /* how long (ms) until repeat */
+ int k_repeat_step; /* how long (ms) until more repeats */
+
+ struct wsconsio_bell_data k_belldata;
+} kbd_softc;
+
+void
+kbdattach(idev, ispec)
+ struct device *idev;
+ struct wscons_idev_spec *ispec;
+{
+ register struct kbd_softc *k = &kbd_softc;
+
+ /*
+ * It would be nice if the repeat rates were in ticks.
+ * However, if they were, we couldn't set them here, as
+ * hz might not be set up yet!
+ */
+ k->k_repeat_start = 200;
+ k->k_repeat_step = 50;
+
+ k->k_belldata.wbd_pitch = 1500; /* 1500 Hz */
+ k->k_belldata.wbd_period = 100; /* 100 ms */
+ k->k_belldata.wbd_volume = 50; /* 50% volume */
+
+ k->k_idev = idev;
+ k->k_ispec = *ispec;
+}
+
+void
+kbd_repeat(void *arg)
+{
+ struct kbd_softc *k = (struct kbd_softc *)arg;
+ int s = spltty();
+
+ if (k->k_repeating && k->k_repeatcp != NULL) {
+ wscons_input(k->k_repeatcp);
+ timeout(kbd_repeat, k, (hz * k->k_repeat_step) / 1000);
+ }
+ splx(s);
+}
+
+void
+kbd_input(register int c)
+{
+ register struct kbd_softc *k = &kbd_softc;
+ register struct firm_event *fe;
+ register int put;
+ char *cp;
+
+ if (k->k_repeating) {
+ k->k_repeating = 0;
+ untimeout(kbd_repeat, k);
+ }
+
+ /*
+ * If /dev/kbd is not connected in event mode translate and
+ * send upstream.
+ */
+ if (!k->k_evmode) {
+ cp = (*k->k_ispec.wi_translate)(k->k_idev, c);
+ if (cp != NULL) {
+ wscons_input(cp);
+ k->k_repeating = 1;
+ k->k_repeatcp = cp;
+ timeout(kbd_repeat, k, (hz * k->k_repeat_start) / 1000);
+ }
+ return;
+ }
+
+ /*
+ * Keyboard is generating events. Turn this keystroke into an
+ * event and put it in the queue. If the queue is full, the
+ * keystroke is lost (sorry!).
+ */
+ put = k->k_events.ev_put;
+ fe = &k->k_events.ev_q[put];
+ put = (put + 1) % EV_QSIZE;
+ if (put == k->k_events.ev_get) {
+ log(LOG_WARNING, "keyboard event queue overflow\n"); /* ??? */
+ return;
+ }
+ fe->id = c & k->k_ispec.wi_keymask;
+ fe->value = (c & k->k_ispec.wi_keyupmask) != 0 ? VKEY_UP : VKEY_DOWN;
+ microtime(&fe->time);
+ k->k_events.ev_put = put;
+ EV_WAKEUP(&k->k_events);
+}
+
+int
+kbdopen(dev_t dev, int flags, int mode, struct proc *p)
+{
+ int s;
+ struct tty *tp;
+
+ if (kbd_softc.k_events.ev_io)
+ return (EBUSY);
+ kbd_softc.k_events.ev_io = p;
+ ev_init(&kbd_softc.k_events);
+ kbd_softc.k_evmode = 1;
+ return (0);
+}
+
+int
+kbdclose(dev_t dev, int flags, int mode, struct proc *p)
+{
+
+ /*
+ * Turn off event mode, dump the queue, and close the keyboard
+ * unless it is supplying console input.
+ */
+ kbd_softc.k_evmode = 0;
+ ev_fini(&kbd_softc.k_events);
+ kbd_softc.k_events.ev_io = NULL;
+ return (0);
+}
+
+int
+kbdread(dev_t dev, struct uio *uio, int flags)
+{
+
+ return (ev_read(&kbd_softc.k_events, uio, flags));
+}
+
+/* this routine should not exist, but is convenient to write here for now */
+int
+kbdwrite(dev_t dev, struct uio *uio, int flags)
+{
+
+ return (EOPNOTSUPP);
+}
+
+int
+kbdioctl(dev_t dev, u_long cmd, register caddr_t data, int flag, struct proc *p)
+{
+ struct kbd_softc *k = &kbd_softc;
+ struct wsconsio_bell_data *wbd;
+ int rv;
+
+ rv = ENOTTY;
+ switch (cmd) {
+#if 0
+ case KIOCSDIRECT:
+ k->k_evmode = *(int *)data;
+ return (0);
+#endif
+
+ case FIONBIO: /* we will remove this someday (soon???) */
+ return (0);
+
+ case FIOASYNC:
+ k->k_events.ev_async = *(int *)data != 0;
+ return (0);
+
+ case TIOCSPGRP:
+ if (*(int *)data != k->k_events.ev_io->p_pgid)
+ return (EPERM);
+ return (0);
+
+ case WSCONSIO_BELL:
+ if (k->k_ispec.wi_bell != NULL)
+ (*k->k_ispec.wi_bell)(k->k_idev, &k->k_belldata);
+ return (0);
+
+ case WSCONSIO_COMPLEXBELL:
+ if (k->k_ispec.wi_bell != NULL) {
+ wbd = (struct wsconsio_bell_data *)data;
+ if ((wbd->wbd_flags & WSCONSIO_BELLDATA_PITCH) == 0)
+ wbd->wbd_pitch = k->k_belldata.wbd_pitch;
+ if ((wbd->wbd_flags & WSCONSIO_BELLDATA_PERIOD) == 0)
+ wbd->wbd_period = k->k_belldata.wbd_period;
+ if ((wbd->wbd_flags & WSCONSIO_BELLDATA_VOLUME) == 0)
+ wbd->wbd_volume = k->k_belldata.wbd_volume;
+
+ (*k->k_ispec.wi_bell)(k->k_idev, wbd);
+ }
+ return (0);
+
+ case WSCONSIO_SETBELL:
+ wbd = (struct wsconsio_bell_data *)data;
+ if ((wbd->wbd_flags & WSCONSIO_BELLDATA_PITCH) != 0)
+ k->k_belldata.wbd_pitch = wbd->wbd_pitch;
+ if ((wbd->wbd_flags & WSCONSIO_BELLDATA_PERIOD) != 0)
+ k->k_belldata.wbd_period = wbd->wbd_period;
+ if ((wbd->wbd_flags & WSCONSIO_BELLDATA_VOLUME) != 0)
+ k->k_belldata.wbd_volume = wbd->wbd_volume;
+ return (0);
+
+ case WSCONSIO_GETBELL:
+ wbd = (struct wsconsio_bell_data *)data;
+ wbd->wbd_flags = WSCONSIO_BELLDATA_PITCH |
+ WSCONSIO_BELLDATA_PERIOD | WSCONSIO_BELLDATA_VOLUME;
+ wbd->wbd_pitch = k->k_belldata.wbd_pitch;
+ wbd->wbd_period = k->k_belldata.wbd_period;
+ wbd->wbd_volume = k->k_belldata.wbd_volume ;
+ return (0);
+
+#if 0 /* XXX */
+ /* XXX KEY-REPEAT RATE SETTING */
+#endif /* XXX */
+
+ default:
+ if (k->k_ispec.wi_ioctl != NULL)
+ rv = (*k->k_ispec.wi_ioctl)(k->k_idev, cmd, data,
+ flag, p);
+ }
+ return (rv);
+}
+
+int
+kbdselect(dev_t dev, int rw, struct proc *p)
+{
+
+ return (ev_select(&kbd_softc.k_events, rw, p));
+}
+
+/* Ring the console bell. (For wscons terminal emulator and other code) */
+void
+wscons_kbd_bell()
+{
+ struct kbd_softc *k = &kbd_softc;
+
+ if (k->k_ispec.wi_bell != NULL)
+ (*k->k_ispec.wi_bell)(k->k_idev, &k->k_belldata);
+}
+
+/*
+ * Console handling functions.
+ */
+
+int
+kbd_cngetc(dev)
+ dev_t dev;
+{
+ struct kbd_softc *k = &kbd_softc;
+
+ if (kbd_softc.k_evmode) /* XXX? */
+ return 0;
+ if (k->k_ispec.wi_getc != NULL)
+ return (*k->k_ispec.wi_getc)(k->k_idev);
+ else
+ return 0;
+}
+
+void
+kbd_cnpollc(dev, on)
+ dev_t dev;
+ int on;
+{
+ struct kbd_softc *k = &kbd_softc;
+
+ if (kbd_softc.k_evmode) /* XXX? */
+ return;
+ if (k->k_ispec.wi_pollc != NULL)
+ (*k->k_ispec.wi_pollc)(k->k_idev, on);
+}