summaryrefslogtreecommitdiff
path: root/sys/arch/sparc64/dev/pcons.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/arch/sparc64/dev/pcons.c')
-rw-r--r--sys/arch/sparc64/dev/pcons.c501
1 files changed, 501 insertions, 0 deletions
diff --git a/sys/arch/sparc64/dev/pcons.c b/sys/arch/sparc64/dev/pcons.c
new file mode 100644
index 00000000000..1c919bf14fb
--- /dev/null
+++ b/sys/arch/sparc64/dev/pcons.c
@@ -0,0 +1,501 @@
+/* $OpenBSD: pcons.c,v 1.1 2001/08/19 05:21:37 jason Exp $ */
+/* $NetBSD: pcons.c,v 1.7 2001/05/02 10:32:20 scw Exp $ */
+
+/*-
+ * Copyright (c) 2000 Eduardo E. Horvath
+ * All rights reserved.
+ *
+ * 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. The name of the author may not be used to endorse or promote products
+ * derived from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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.
+ */
+
+/*
+ * Default console driver. Uses the PROM or whatever
+ * driver(s) are appropriate.
+ */
+
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/conf.h>
+#include <sys/device.h>
+#include <sys/file.h>
+#include <sys/ioctl.h>
+#include <sys/kernel.h>
+#include <sys/proc.h>
+#include <sys/tty.h>
+#include <sys/time.h>
+#include <sys/syslog.h>
+
+#include <machine/autoconf.h>
+#include <machine/openfirm.h>
+#include <machine/bsd_openprom.h>
+#include <machine/conf.h>
+#include <machine/cpu.h>
+#include <machine/eeprom.h>
+#include <machine/psl.h>
+
+#include <dev/cons.h>
+
+/* XXXXXXXX - this is in MI code in NetBSD */
+/*
+ * Stuff to handle debugger magic key sequences.
+ */
+#define CNS_LEN 128
+#define CNS_MAGIC_VAL(x) ((x)&0x1ff)
+#define CNS_MAGIC_NEXT(x) (((x)>>9)&0x7f)
+#define CNS_TERM 0x7f /* End of sequence */
+
+typedef struct cnm_state {
+ int cnm_state;
+ u_short *cnm_magic;
+} cnm_state_t;
+extern int db_console;
+#ifdef DDB
+#define cn_trap() do { if (db_console) Debugger(); } while (0)
+#else
+#define cn_trap()
+#endif
+#define cn_isconsole(d) ((d) == cn_tab->cn_dev)
+void cn_init_magic __P((cnm_state_t *cnm));
+void cn_destroy_magic __P((cnm_state_t *cnm));
+int cn_set_magic __P((char *magic));
+int cn_get_magic __P((char *magic, int len));
+/* This should be called for each byte read */
+#ifndef cn_check_magic
+#define cn_check_magic(d, k, s) \
+ do { \
+ if (cn_isconsole(d)) { \
+ int v = (s).cnm_magic[(s).cnm_state]; \
+ if ((k) == CNS_MAGIC_VAL(v)) { \
+ (s).cnm_state = CNS_MAGIC_NEXT(v); \
+ if ((s).cnm_state == CNS_TERM) { \
+ cn_trap(); \
+ (s).cnm_state = 0; \
+ } \
+ } else { \
+ (s).cnm_state = 0; \
+ } \
+ } \
+ } while (/* CONSTCOND */ 0)
+#endif
+
+/* Encode out-of-band events this way when passing to cn_check_magic() */
+#define CNC_BREAK 0x100
+
+/* XXXXXXXXXX - end of this part of cnmagic, more at the end of this file. */
+
+#include <sparc64/dev/cons.h>
+
+static int pconsmatch __P((struct device *, void *, void *));
+static void pconsattach __P((struct device *, struct device *, void *));
+
+struct cfattach pcons_ca = {
+ sizeof(struct pconssoftc), pconsmatch, pconsattach
+};
+
+struct cfdriver pcons_cd = {
+ NULL, "pcons", DV_TTY
+};
+
+extern struct cfdriver pcons_cd;
+static struct cnm_state pcons_cnm_state;
+
+static int pconsprobe __P((void));
+extern struct consdev *cn_tab;
+
+static int
+pconsmatch(parent, match, aux)
+ struct device *parent;
+ void *match;
+ void *aux;
+{
+ struct mainbus_attach_args *ma = aux;
+ extern int prom_cngetc __P((dev_t));
+
+ /* Only attach if no other console has attached. */
+ return ((strcmp("pcons", ma->ma_name) == 0) &&
+ (cn_tab->cn_getc == prom_cngetc));
+
+}
+
+static void pcons_poll __P((void *));
+
+static void
+pconsattach(parent, self, aux)
+ struct device *parent, *self;
+ void *aux;
+{
+ struct pconssoftc *sc = (struct pconssoftc *) self;
+
+ printf("\n");
+ if (!pconsprobe())
+ return;
+
+ cn_init_magic(&pcons_cnm_state);
+ cn_set_magic("+++++");
+ timeout_set(&sc->sc_poll_to, pcons_poll, sc);
+}
+
+static void pconsstart __P((struct tty *));
+static int pconsparam __P((struct tty *, struct termios *));
+
+int
+pconsopen(dev, flag, mode, p)
+ dev_t dev;
+ int flag, mode;
+ struct proc *p;
+{
+ struct pconssoftc *sc;
+ int unit = minor(dev);
+ struct tty *tp;
+
+ if (unit >= pcons_cd.cd_ndevs)
+ return ENXIO;
+ sc = pcons_cd.cd_devs[unit];
+ if (!sc)
+ return ENXIO;
+ if (!(tp = sc->of_tty))
+ sc->of_tty = tp = ttymalloc();
+ tp->t_oproc = pconsstart;
+ tp->t_param = pconsparam;
+ tp->t_dev = dev;
+ cn_tab->cn_dev = dev;
+ if (!(tp->t_state & TS_ISOPEN)) {
+ ttychars(tp);
+ tp->t_iflag = TTYDEF_IFLAG;
+ tp->t_oflag = TTYDEF_OFLAG;
+ tp->t_cflag = TTYDEF_CFLAG;
+ tp->t_lflag = TTYDEF_LFLAG;
+ tp->t_ispeed = tp->t_ospeed = TTYDEF_SPEED;
+ pconsparam(tp, &tp->t_termios);
+ ttsetwater(tp);
+ } else if ((tp->t_state&TS_XCLUDE) && suser(p->p_ucred, &p->p_acflag))
+ return EBUSY;
+ tp->t_state |= TS_CARR_ON;
+
+ if (!(sc->of_flags & OFPOLL)) {
+ sc->of_flags |= OFPOLL;
+ timeout_add(&sc->sc_poll_to, 1);
+ }
+
+ return (*linesw[tp->t_line].l_open)(dev, tp);
+}
+
+int
+pconsclose(dev, flag, mode, p)
+ dev_t dev;
+ int flag, mode;
+ struct proc *p;
+{
+ struct pconssoftc *sc = pcons_cd.cd_devs[minor(dev)];
+ struct tty *tp = sc->of_tty;
+
+ timeout_del(&sc->sc_poll_to);
+ sc->of_flags &= ~OFPOLL;
+ (*linesw[tp->t_line].l_close)(tp, flag);
+ ttyclose(tp);
+ return 0;
+}
+
+int
+pconsread(dev, uio, flag)
+ dev_t dev;
+ struct uio *uio;
+ int flag;
+{
+ struct pconssoftc *sc = pcons_cd.cd_devs[minor(dev)];
+ struct tty *tp = sc->of_tty;
+
+ return (*linesw[tp->t_line].l_read)(tp, uio, flag);
+}
+
+int
+pconswrite(dev, uio, flag)
+ dev_t dev;
+ struct uio *uio;
+ int flag;
+{
+ struct pconssoftc *sc = pcons_cd.cd_devs[minor(dev)];
+ struct tty *tp = sc->of_tty;
+
+ return (*linesw[tp->t_line].l_write)(tp, uio, flag);
+}
+
+int
+pconsioctl(dev, cmd, data, flag, p)
+ dev_t dev;
+ u_long cmd;
+ caddr_t data;
+ int flag;
+ struct proc *p;
+{
+ struct pconssoftc *sc = pcons_cd.cd_devs[minor(dev)];
+ struct tty *tp = sc->of_tty;
+ int error;
+
+ if ((error = (*linesw[tp->t_line].l_ioctl)(tp, cmd, data, flag, p)) >= 0)
+ return error;
+ if ((error = ttioctl(tp, cmd, data, flag, p)) >= 0)
+ return error;
+ return ENOTTY;
+}
+
+struct tty *
+pconstty(dev)
+ dev_t dev;
+{
+ struct pconssoftc *sc = pcons_cd.cd_devs[minor(dev)];
+
+ return sc->of_tty;
+}
+
+int
+pconsstop(tp, flag)
+ struct tty *tp;
+ int flag;
+{
+ return 0;
+}
+
+static void
+pconsstart(tp)
+ struct tty *tp;
+{
+ struct clist *cl;
+ int s, len;
+ u_char buf[OFBURSTLEN];
+
+ s = spltty();
+ if (tp->t_state & (TS_TIMEOUT | TS_BUSY | TS_TTSTOP)) {
+ splx(s);
+ return;
+ }
+ tp->t_state |= TS_BUSY;
+ splx(s);
+ cl = &tp->t_outq;
+ len = q_to_b(cl, buf, OFBURSTLEN);
+ OF_write(stdout, buf, len);
+ s = spltty();
+ tp->t_state &= ~TS_BUSY;
+ if (cl->c_cc) {
+ tp->t_state |= TS_TIMEOUT;
+ timeout_add(&tp->t_rstrt_to, 1);
+ }
+ if (cl->c_cc <= tp->t_lowat) {
+ if (tp->t_state & TS_ASLEEP) {
+ tp->t_state &= ~TS_ASLEEP;
+ wakeup(cl);
+ }
+ selwakeup(&tp->t_wsel);
+ }
+ splx(s);
+}
+
+static int
+pconsparam(tp, t)
+ struct tty *tp;
+ struct termios *t;
+{
+ tp->t_ispeed = t->c_ispeed;
+ tp->t_ospeed = t->c_ospeed;
+ tp->t_cflag = t->c_cflag;
+ return 0;
+}
+
+static void
+pcons_poll(aux)
+ void *aux;
+{
+ struct pconssoftc *sc = aux;
+ struct tty *tp = sc->of_tty;
+ char ch;
+
+ while (OF_read(stdin, &ch, 1) > 0) {
+ cn_check_magic(tp->t_dev, ch, pcons_cnm_state);
+ if (tp && (tp->t_state & TS_ISOPEN))
+ (*linesw[tp->t_line].l_rint)(ch, tp);
+ }
+ timeout_add(&sc->sc_poll_to, 1);
+}
+
+int
+pconsprobe()
+{
+ if (!stdin) stdin = OF_stdin();
+ if (!stdout) stdout = OF_stdout();
+
+ return (stdin && stdout);
+}
+
+void
+pcons_cnpollc(dev, on)
+ dev_t dev;
+ int on;
+{
+ struct pconssoftc *sc = NULL;
+
+ if (pcons_cd.cd_devs)
+ sc = pcons_cd.cd_devs[minor(dev)];
+
+ if (on) {
+ if (!sc) return;
+ if (sc->of_flags & OFPOLL)
+ timeout_del(&sc->sc_poll_to);
+ sc->of_flags &= ~OFPOLL;
+ } else {
+ /* Resuming kernel. */
+ if (sc && !(sc->of_flags & OFPOLL)) {
+ sc->of_flags |= OFPOLL;
+ timeout_add(&sc->sc_poll_to, 1);
+ }
+ }
+}
+
+void pcons_dopoll __P((void));
+void
+pcons_dopoll() {
+ pcons_poll((void*)pcons_cd.cd_devs[0]);
+}
+
+/* XXXXXXXX --- more cnmagic stuff. */
+#define ENCODE_STATE(c, n) (short)(((c)&0x1ff)|(((n)&0x7f)<<9))
+
+static unsigned short cn_magic[CNS_LEN];
+
+/*
+ * Initialize a cnm_state_t.
+ */
+void
+cn_init_magic(cnm_state_t *cnm)
+{
+ cnm->cnm_state = 0;
+ cnm->cnm_magic = cn_magic;
+}
+
+/*
+ * Destroy a cnm_state_t.
+ */
+void
+cn_destroy_magic(cnm_state_t *cnm)
+{
+ cnm->cnm_state = 0;
+ cnm->cnm_magic = NULL;
+}
+
+/*
+ * Translate a magic string to a state
+ * machine table.
+ */
+int
+cn_set_magic(char *magic)
+{
+ unsigned int i, c, n;
+ unsigned short m[CNS_LEN];
+
+ for (i=0; i<CNS_LEN; i++) {
+ c = (*magic++)&0xff;
+ n = *magic ? i+1 : CNS_TERM;
+ switch (c) {
+ case 0:
+ /* End of string */
+ if (i == 0) {
+ /* empty string? */
+ cn_magic[0] = 0;
+#ifdef DEBUG
+ printf("cn_set_magic(): empty!\n");
+#endif
+ return (0);
+ }
+ do {
+ cn_magic[i] = m[i];
+ } while (i--);
+ return(0);
+ case 0x27:
+ /* Escape sequence */
+ c = (*magic++)&0xff;
+ n = *magic ? i+1 : CNS_TERM;
+ switch (c) {
+ case 0x27:
+ break;
+ case 0x01:
+ /* BREAK */
+ c = CNC_BREAK;
+ break;
+ case 0x02:
+ /* NUL */
+ c = 0;
+ break;
+ }
+ /* FALLTHROUGH */
+ default:
+ /* Transition to the next state. */
+#ifdef DEBUG
+ if (!cold)
+ printf("mag %d %x:%x\n", i, c, n);
+#endif
+ m[i] = ENCODE_STATE(c, n);
+ break;
+ }
+ }
+ return (EINVAL);
+}
+
+/*
+ * Translatea state machine table back to
+ * a magic string.
+ */
+int
+cn_get_magic(char *magic, int maglen) {
+ unsigned int i, c;
+
+ for (i=0; i<CNS_LEN; i++) {
+ c = cn_magic[i];
+ /* Translate a character */
+ switch (CNS_MAGIC_VAL(c)) {
+ case CNC_BREAK:
+ *magic++ = 0x27;
+ *magic++ = 0x01;
+ break;
+ case 0:
+ *magic++ = 0x27;
+ *magic++ = 0x02;
+ break;
+ case 0x27:
+ *magic++ = 0x27;
+ *magic++ = 0x27;
+ break;
+ default:
+ *magic++ = (c&0x0ff);
+ break;
+ }
+ /* Now go to the next state */
+ i = CNS_MAGIC_NEXT(c);
+ if (i == CNS_TERM || i == 0) {
+ /* Either termination state or empty machine */
+ *magic++ = 0;
+ return (0);
+ }
+ }
+ return (EINVAL);
+}
+