/* $OpenBSD: vcons.c,v 1.2 2008/07/07 14:38:42 kettenis Exp $ */ /* * Copyright (c) 2008 Mark Kettenis * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ #include #include #include #include #include #include #include #ifdef DDB #include #endif #include #include #include #include #include #include struct vcons_softc { struct device sc_dev; struct tty *sc_tty; struct timeout sc_to; }; int vcons_match(struct device *, void *, void *); void vcons_attach(struct device *, struct device *, void *); struct cfattach vcons_ca = { sizeof(struct vcons_softc), vcons_match, vcons_attach }; struct cfdriver vcons_cd = { NULL, "vcons", DV_DULL }; int vcons_cnlookc(dev_t, int *); int vcons_cngetc(dev_t); void vcons_cnputc(dev_t, int); void vconsstart(struct tty *); int vconsparam(struct tty *, struct termios *); void vconstimeout(void *); int vcons_match(struct device *parent, void *match, void *aux) { struct vbus_attach_args *va = aux; if (strcmp(va->va_name, "console") == 0) return (1); return (0); } void vcons_attach(struct device *parent, struct device *self, void *aux) { struct vcons_softc *sc = (struct vcons_softc *)self; int maj; printf("\n"); timeout_set(&sc->sc_to, vconstimeout, sc); cn_tab->cn_pollc = nullcnpollc; cn_tab->cn_getc = vcons_cngetc; cn_tab->cn_putc = vcons_cnputc; /* Locate the major number. */ for (maj = 0; maj < nchrdev; maj++) if (cdevsw[maj].d_open == vconsopen) break; cn_tab->cn_dev = makedev(maj, self->dv_unit); } int vcons_cnlookc(dev_t dev, int *cp) { int64_t ch; if (hv_cons_getchar(&ch) == H_EOK) { #ifdef DDB if (ch == -1 && db_console) Debugger(); #endif *cp = ch; return (1); } return (0); } int vcons_cngetc(dev_t dev) { int c; while(!vcons_cnlookc(dev, &c)) ; return (c); } void vcons_cnputc(dev_t dev, int c) { while (hv_cons_putchar(c) == H_EWOULDBLOCK); } int vconsopen(dev_t dev, int flag, int mode, struct proc *p) { struct vcons_softc *sc; struct tty *tp; int unit = minor(dev); int error, setuptimeout; if (unit > vcons_cd.cd_ndevs) return (ENXIO); sc = vcons_cd.cd_devs[unit]; if (sc == NULL) return (ENXIO); if (sc->sc_tty) tp = sc->sc_tty; else tp = sc->sc_tty = ttymalloc(); tp->t_oproc = vconsstart; tp->t_param = vconsparam; tp->t_dev = dev; if ((tp->t_state & TS_ISOPEN) == 0) { 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; ttsetwater(tp); setuptimeout = 1; } else if ((tp->t_state & TS_XCLUDE) && suser(p, 0)) return (EBUSY); tp->t_state |= TS_CARR_ON; error = (*linesw[tp->t_line].l_open)(dev, tp); if (error == 0 && setuptimeout) vconstimeout(sc); return (error); } int vconsclose(dev_t dev, int flag, int mode, struct proc *p) { struct vcons_softc *sc; struct tty *tp; int unit = minor(dev); if (unit > vcons_cd.cd_ndevs) return (ENXIO); sc = vcons_cd.cd_devs[unit]; if (sc == NULL) return (ENXIO); tp = sc->sc_tty; timeout_del(&sc->sc_to); (*linesw[tp->t_line].l_close)(tp, flag); ttyclose(tp); return (0); } int vconsread(dev_t dev, struct uio *uio, int flag) { struct vcons_softc *sc; struct tty *tp; int unit = minor(dev); if (unit > vcons_cd.cd_ndevs) return (ENXIO); sc = vcons_cd.cd_devs[unit]; if (sc == NULL) return (ENXIO); tp = sc->sc_tty; return ((*linesw[tp->t_line].l_read)(tp, uio, flag)); } int vconswrite(dev_t dev, struct uio *uio, int flag) { struct vcons_softc *sc; struct tty *tp; int unit = minor(dev); if (unit > vcons_cd.cd_ndevs) return (ENXIO); sc = vcons_cd.cd_devs[unit]; if (sc == NULL) return (ENXIO); tp = sc->sc_tty; return ((*linesw[tp->t_line].l_write)(tp, uio, flag)); } int vconsioctl(dev_t dev, u_long cmd, caddr_t data, int flag, struct proc *p) { struct vcons_softc *sc; struct tty *tp; int unit = minor(dev); int error; if (unit > vcons_cd.cd_ndevs) return (ENXIO); sc = vcons_cd.cd_devs[unit]; if (sc == NULL) return (ENXIO); tp = sc->sc_tty; error = (*linesw[tp->t_line].l_ioctl)(tp, cmd, data, flag, p); if (error >= 0) return error; error = ttioctl(tp, cmd, data, flag, p); if (error >= 0) return (error); return (ENOTTY); } void vconsstart(struct tty *tp) { int s; s = spltty(); if (tp->t_state & (TS_TTSTOP | TS_BUSY)) { splx(s); return; } if (tp->t_outq.c_cc <= tp->t_lowat) { if (tp->t_state & TS_ASLEEP) { tp->t_state &= ~TS_ASLEEP; wakeup((caddr_t)&tp->t_outq); } selwakeup(&tp->t_wsel); } tp->t_state |= TS_BUSY; while (tp->t_outq.c_cc != 0) vcons_cnputc(tp->t_dev, getc(&tp->t_outq)); tp->t_state &= ~TS_BUSY; splx(s); } int vconsstop(struct tty *tp, int flag) { int s; s = spltty(); if (tp->t_state & TS_BUSY) if ((tp->t_state & TS_TTSTOP) == 0) tp->t_state |= TS_FLUSH; splx(s); return (0); } struct tty * vconstty(dev_t dev) { struct vcons_softc *sc; int unit = minor(dev); if (unit > vcons_cd.cd_ndevs) return (NULL); sc = vcons_cd.cd_devs[unit]; if (sc == NULL) return (NULL); return sc->sc_tty; } int vconsparam(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); } void vconstimeout(void *v) { struct vcons_softc *sc = v; struct tty *tp = sc->sc_tty; int c; while (vcons_cnlookc(tp->t_dev, &c)) { if (tp->t_state & TS_ISOPEN) (*linesw[tp->t_line].l_rint)(c, tp); } timeout_add(&sc->sc_to, 1); }