freebsd-dev/sys/alpha/tlsb/zs_tlsb.c

474 lines
9.2 KiB
C
Raw Normal View History

/*-
* 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);