474 lines
9.2 KiB
C
474 lines
9.2 KiB
C
|
/*-
|
||
|
* Copyright (c) 1998 Doug Rabson
|
||
|
* 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.
|
||
|
*
|
||
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR 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 AUTHOR 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.
|
||
|
*
|
||
|
* $Id$
|
||
|
*/
|
||
|
/*
|
||
|
* This driver is a hopeless hack to get the SimOS console working. A real
|
||
|
* driver would use the zs driver source from NetBSD.
|
||
|
*/
|
||
|
|
||
|
#include "opt_ddb.h"
|
||
|
|
||
|
#include <sys/param.h>
|
||
|
#include <sys/kernel.h>
|
||
|
#include <sys/systm.h>
|
||
|
#include <sys/module.h>
|
||
|
#include <sys/bus.h>
|
||
|
#include <sys/conf.h>
|
||
|
#include <sys/tty.h>
|
||
|
#include <sys/proc.h>
|
||
|
#include <sys/ucred.h>
|
||
|
#include <machine/cons.h>
|
||
|
#include <machine/clock.h>
|
||
|
|
||
|
#include <alpha/tlsb/gbusvar.h>
|
||
|
#include <alpha/tlsb/tlsbreg.h> /* XXX */
|
||
|
#include <alpha/tlsb/zsreg.h>
|
||
|
|
||
|
#define KV(_addr) ((caddr_t)ALPHA_PHYS_TO_K0SEG((_addr)))
|
||
|
|
||
|
static int zsc_get_channel(device_t dev);
|
||
|
static caddr_t zsc_get_base(device_t dev);
|
||
|
|
||
|
struct zs_softc {
|
||
|
struct tty tty;
|
||
|
int channel;
|
||
|
caddr_t base;
|
||
|
};
|
||
|
#define ZS_SOFTC(unit) \
|
||
|
((struct zs_softc*)devclass_get_softc(zs_devclass, (unit)))
|
||
|
|
||
|
static d_open_t zsopen;
|
||
|
static d_close_t zsclose;
|
||
|
static d_read_t zsread;
|
||
|
static d_write_t zswrite;
|
||
|
static d_ioctl_t zsioctl;
|
||
|
static d_stop_t zsstop;
|
||
|
static d_devtotty_t zsdevtotty;
|
||
|
|
||
|
#define CDEV_MAJOR 97
|
||
|
static struct cdevsw zs_cdevsw = {
|
||
|
zsopen, zsclose, zsread, zswrite,
|
||
|
zsioctl, zsstop, noreset, zsdevtotty,
|
||
|
ttpoll, nommap, NULL, "zs",
|
||
|
NULL, -1,
|
||
|
};
|
||
|
|
||
|
static void zsstart __P((struct tty *));
|
||
|
static int zsparam __P((struct tty *, struct termios *));
|
||
|
|
||
|
/*
|
||
|
* Helpers for console support.
|
||
|
*/
|
||
|
|
||
|
static int zs_cngetc __P((dev_t));
|
||
|
static void zs_cnputc __P((dev_t, int));
|
||
|
static void zs_cnpollc __P((dev_t, int));
|
||
|
|
||
|
struct consdev zs_cons = {
|
||
|
NULL, NULL, zs_cngetc, NULL, zs_cnputc,
|
||
|
NULL, makedev(CDEV_MAJOR, 0), CN_NORMAL,
|
||
|
};
|
||
|
|
||
|
static caddr_t zs_console_addr;
|
||
|
static int zs_console;
|
||
|
|
||
|
static int zs_probe(bus_t, device_t);
|
||
|
static int zs_attach(bus_t, device_t);
|
||
|
|
||
|
static devclass_t zs_devclass;
|
||
|
static devclass_t zsc_devclass;
|
||
|
|
||
|
driver_t zs_driver = {
|
||
|
"zs",
|
||
|
zs_probe,
|
||
|
zs_attach,
|
||
|
NULL,
|
||
|
NULL,
|
||
|
DRIVER_TYPE_MISC,
|
||
|
sizeof(struct zs_softc),
|
||
|
NULL,
|
||
|
};
|
||
|
|
||
|
static int
|
||
|
zs_probe(bus_t bus, device_t dev)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
zs_attach(bus_t bus, device_t dev)
|
||
|
{
|
||
|
struct zs_softc *sc = device_get_softc(dev);
|
||
|
|
||
|
sc->channel = zsc_get_channel(dev);
|
||
|
sc->base = zsc_get_base(dev);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
zs_get_status(caddr_t base)
|
||
|
{
|
||
|
return (*(u_int32_t*) (base + ZSC_STATUS)) & 0xff;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
zs_put_status(caddr_t base, int v)
|
||
|
{
|
||
|
*(u_int32_t*) (base + ZSC_STATUS) = v;
|
||
|
alpha_mb();
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
zs_get_rr3(caddr_t base)
|
||
|
{
|
||
|
zs_put_status(base, 3);
|
||
|
return zs_get_status(base);
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
zs_get_data(caddr_t base)
|
||
|
{
|
||
|
return (*(u_int32_t*) (base + ZSC_DATA)) & 0xff;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
zs_put_data(caddr_t base, int v)
|
||
|
{
|
||
|
*(u_int32_t*) (base + ZSC_DATA) = v;
|
||
|
alpha_mb();
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
zs_getc(caddr_t base)
|
||
|
{
|
||
|
while (!(zs_get_status(base) & 1))
|
||
|
DELAY(5);
|
||
|
return zs_get_data(base);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
zs_putc(caddr_t base, int c)
|
||
|
{
|
||
|
while (!(zs_get_status(base) & 4))
|
||
|
DELAY(5);
|
||
|
zs_put_data(base, c);
|
||
|
}
|
||
|
|
||
|
extern struct consdev* cn_tab;
|
||
|
|
||
|
int
|
||
|
zs_cnattach(vm_offset_t base, vm_offset_t offset)
|
||
|
{
|
||
|
zs_console_addr = (caddr_t) ALPHA_PHYS_TO_K0SEG(base + offset);
|
||
|
zs_console = 1;
|
||
|
|
||
|
cn_tab = &zs_cons;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
zs_cngetc(dev_t dev)
|
||
|
{
|
||
|
int s = spltty();
|
||
|
int c = zs_getc(zs_console_addr);
|
||
|
splx(s);
|
||
|
return c;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
zs_cnputc(dev_t dev, int c)
|
||
|
{
|
||
|
int s = spltty();
|
||
|
zs_putc(zs_console_addr, c);
|
||
|
splx(s);
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
zs_cnpollc(dev_t dev, int onoff)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
|
||
|
static int
|
||
|
zsopen(dev_t dev, int flag, int mode, struct proc *p)
|
||
|
{
|
||
|
struct zs_softc* sc = ZS_SOFTC(minor(dev));
|
||
|
struct tty *tp;
|
||
|
int s;
|
||
|
int error = 0;
|
||
|
|
||
|
if (!sc)
|
||
|
return ENXIO;
|
||
|
|
||
|
s = spltty();
|
||
|
|
||
|
tp = &sc->tty;
|
||
|
|
||
|
tp->t_oproc = zsstart;
|
||
|
tp->t_param = zsparam;
|
||
|
tp->t_dev = dev;
|
||
|
if ((tp->t_state & TS_ISOPEN) == 0) {
|
||
|
tp->t_state |= TS_CARR_ON;
|
||
|
ttychars(tp);
|
||
|
tp->t_iflag = TTYDEF_IFLAG;
|
||
|
tp->t_oflag = TTYDEF_OFLAG;
|
||
|
tp->t_cflag = TTYDEF_CFLAG|CLOCAL;
|
||
|
tp->t_lflag = TTYDEF_LFLAG;
|
||
|
tp->t_ispeed = tp->t_ospeed = 9600;
|
||
|
ttsetwater(tp);
|
||
|
} else if (tp->t_state&TS_XCLUDE && p->p_ucred->cr_uid != 0) {
|
||
|
splx(s);
|
||
|
return EBUSY;
|
||
|
}
|
||
|
|
||
|
splx(s);
|
||
|
|
||
|
error = (*linesw[tp->t_line].l_open)(dev, tp);
|
||
|
|
||
|
return error;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
zsclose(dev_t dev, int flag, int mode, struct proc *p)
|
||
|
{
|
||
|
struct tty *tp = &ZS_SOFTC(minor(dev))->tty;
|
||
|
|
||
|
(*linesw[tp->t_line].l_close)(tp, flag);
|
||
|
ttyclose(tp);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
zsread(dev_t dev, struct uio *uio, int flag)
|
||
|
{
|
||
|
struct tty *tp = &ZS_SOFTC(minor(dev))->tty;
|
||
|
return ((*linesw[tp->t_line].l_read)(tp, uio, flag));
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
zswrite(dev_t dev, struct uio *uio, int flag)
|
||
|
{
|
||
|
struct tty *tp = &ZS_SOFTC(minor(dev))->tty;
|
||
|
return ((*linesw[tp->t_line].l_write)(tp, uio, flag));
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
zsioctl(dev_t dev, u_long cmd, caddr_t data, int flag, struct proc *p)
|
||
|
{
|
||
|
struct tty *tp = &ZS_SOFTC(minor(dev))->tty;
|
||
|
int error;
|
||
|
|
||
|
error = (*linesw[tp->t_line].l_ioctl)(tp, cmd, data, flag, p);
|
||
|
if (error != ENOIOCTL)
|
||
|
return error;
|
||
|
error = ttioctl(tp, cmd, data, flag);
|
||
|
if (error != ENOIOCTL)
|
||
|
return error;
|
||
|
|
||
|
return ENOTTY;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
zsparam(struct tty *tp, struct termios *t)
|
||
|
{
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
zsstart(struct tty *tp)
|
||
|
{
|
||
|
struct zs_softc* sc = (struct zs_softc*) tp;
|
||
|
int s;
|
||
|
|
||
|
s = spltty();
|
||
|
|
||
|
if (tp->t_state & (TS_TIMEOUT | TS_TTSTOP)) {
|
||
|
ttwwakeup(tp);
|
||
|
splx(s);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
tp->t_state |= TS_BUSY;
|
||
|
while (tp->t_outq.c_cc != 0)
|
||
|
zs_putc(sc->base, getc(&tp->t_outq));
|
||
|
tp->t_state &= ~TS_BUSY;
|
||
|
|
||
|
ttwwakeup(tp);
|
||
|
splx(s);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Stop output on a line.
|
||
|
*/
|
||
|
static void
|
||
|
zsstop(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);
|
||
|
}
|
||
|
|
||
|
static struct tty *
|
||
|
zsdevtotty(dev_t dev)
|
||
|
{
|
||
|
struct zs_softc* sc = ZS_SOFTC(minor(dev));
|
||
|
if (!sc)
|
||
|
return (NULL);
|
||
|
return (&sc->tty);
|
||
|
}
|
||
|
|
||
|
CDEV_DRIVER_MODULE(zs, zsc, zs_driver, zs_devclass,
|
||
|
CDEV_MAJOR, zs_cdevsw, 0, 0);
|
||
|
|
||
|
/*
|
||
|
* The zsc bus holds two zs devices, one for channel A, one for channel B.
|
||
|
*/
|
||
|
|
||
|
struct zsc_softc {
|
||
|
struct bus bus;
|
||
|
caddr_t base;
|
||
|
struct zs_softc* sc_a;
|
||
|
struct zs_softc* sc_b;
|
||
|
};
|
||
|
|
||
|
static driver_probe_t zsc_tlsb_probe;
|
||
|
static driver_attach_t zsc_tlsb_attach;
|
||
|
static driver_intr_t zsc_tlsb_intr;
|
||
|
static bus_print_device_t zsc_tlsb_print_device;
|
||
|
|
||
|
driver_t zsc_tlsb_driver = {
|
||
|
"zsc",
|
||
|
zsc_tlsb_probe,
|
||
|
zsc_tlsb_attach,
|
||
|
NULL,
|
||
|
NULL,
|
||
|
DRIVER_TYPE_MISC,
|
||
|
sizeof(struct zsc_softc),
|
||
|
NULL,
|
||
|
};
|
||
|
|
||
|
|
||
|
static bus_ops_t zsc_tlsb_ops = {
|
||
|
zsc_tlsb_print_device,
|
||
|
null_read_ivar,
|
||
|
null_write_ivar,
|
||
|
null_map_intr,
|
||
|
};
|
||
|
|
||
|
static int
|
||
|
zsc_get_channel(device_t dev)
|
||
|
{
|
||
|
return (long) device_get_ivars(dev);
|
||
|
}
|
||
|
|
||
|
static caddr_t
|
||
|
zsc_get_base(device_t dev)
|
||
|
{
|
||
|
device_t busdev = bus_get_device(device_get_parent(dev));
|
||
|
struct zsc_softc* sc = device_get_softc(busdev);
|
||
|
return sc->base;
|
||
|
}
|
||
|
|
||
|
static void
|
||
|
zsc_tlsb_print_device(bus_t bus, device_t dev)
|
||
|
{
|
||
|
device_t busdev = bus_get_device(bus);
|
||
|
|
||
|
printf(" at %s%d channel %c",
|
||
|
device_get_name(busdev), device_get_unit(busdev),
|
||
|
'A' + (device_get_unit(dev) & 1));
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
zsc_tlsb_probe(bus_t parent, device_t dev)
|
||
|
{
|
||
|
struct zsc_softc* sc = device_get_softc(dev);
|
||
|
|
||
|
device_set_desc(dev, "Z8530 uart");
|
||
|
|
||
|
bus_init(&sc->bus, dev, &zsc_tlsb_ops);
|
||
|
|
||
|
sc->base = (caddr_t) ALPHA_PHYS_TO_K0SEG(TLSB_GBUS_BASE
|
||
|
+ gbus_get_offset(dev));
|
||
|
|
||
|
/*
|
||
|
* Add channel A for now.
|
||
|
*/
|
||
|
bus_add_device(&sc->bus, "zs", -1, (void*) 0);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
zsc_tlsb_attach(bus_t parent, device_t dev)
|
||
|
{
|
||
|
struct zsc_softc* sc = device_get_softc(dev);
|
||
|
|
||
|
bus_generic_attach(parent, dev);
|
||
|
|
||
|
/* XXX */
|
||
|
sc->sc_a = ZS_SOFTC(0);
|
||
|
|
||
|
bus_map_intr(parent, dev, zsc_tlsb_intr, sc);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
zsc_tlsb_intr(void* arg)
|
||
|
{
|
||
|
struct zsc_softc* sc = arg;
|
||
|
caddr_t base = sc->base;
|
||
|
|
||
|
/* XXX only allow for zs0 at zsc0 */
|
||
|
int rr3 = zs_get_rr3(base);
|
||
|
if (rr3 & 0x20) {
|
||
|
struct tty* tp = &sc->sc_a->tty;
|
||
|
int c;
|
||
|
|
||
|
while (zs_get_status(base) & 1) {
|
||
|
c = zs_get_data(base);
|
||
|
#ifdef DDB
|
||
|
if (c == CTRL('\\'))
|
||
|
Debugger("manual escape to debugger");
|
||
|
#endif
|
||
|
if (tp->t_state & TS_ISOPEN)
|
||
|
(*linesw[tp->t_line].l_rint)(c, tp);
|
||
|
DELAY(5);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
DRIVER_MODULE(zsc_tlsb, gbus, zsc_tlsb_driver, zsc_devclass, 0, 0);
|