diff options
author | Miod Vallat <miod@cvs.openbsd.org> | 2004-06-20 21:28:07 +0000 |
---|---|---|
committer | Miod Vallat <miod@cvs.openbsd.org> | 2004-06-20 21:28:07 +0000 |
commit | d2027a358165d1a616aae1c4a7137b94ff0c7ac6 (patch) | |
tree | 736b54f22a6750e7397e0d01e9075655a36c156c | |
parent | c967fc5433b4ef6dba0cc4729567fefc1f83b64b (diff) |
Port mgx(4) to sparc64.
-rw-r--r-- | sys/arch/sparc64/conf/GENERIC | 5 | ||||
-rw-r--r-- | sys/dev/sbus/files.sbus | 6 | ||||
-rw-r--r-- | sys/dev/sbus/mgx.c | 439 |
3 files changed, 448 insertions, 2 deletions
diff --git a/sys/arch/sparc64/conf/GENERIC b/sys/arch/sparc64/conf/GENERIC index fdad36f8102..e3b9de40407 100644 --- a/sys/arch/sparc64/conf/GENERIC +++ b/sys/arch/sparc64/conf/GENERIC @@ -1,4 +1,4 @@ -# $OpenBSD: GENERIC,v 1.94 2004/06/20 20:49:56 miod Exp $ +# $OpenBSD: GENERIC,v 1.95 2004/06/20 21:28:04 miod Exp $ # $NetBSD: GENERIC32,v 1.18 2001/07/20 00:07:12 eeh Exp $ machine sparc64 @@ -273,6 +273,9 @@ wsdisplay* at zx? rfx* at sbus? # RasterFlex framebuffer series wsdisplay* at rfx? +mgx* at sbus? # SMS MGX/MGXPlus framebuffer +wsdisplay* at mgx? + audiocs* at sbus? asio* at sbus? # Aurora 210SJ diff --git a/sys/dev/sbus/files.sbus b/sys/dev/sbus/files.sbus index 47df6b833be..4a1415c8410 100644 --- a/sys/dev/sbus/files.sbus +++ b/sys/dev/sbus/files.sbus @@ -1,4 +1,4 @@ -# $OpenBSD: files.sbus,v 1.25 2004/06/20 20:49:57 miod Exp $ +# $OpenBSD: files.sbus,v 1.26 2004/06/20 21:28:06 miod Exp $ # $NetBSD: files.sbus,v 1.16 2000/12/08 17:29:12 martin Exp $ # # Config file and device description for machine-independent SBUS code. @@ -72,6 +72,10 @@ device rfx: wsemuldisplaydev, rasops8, wsemul_sun attach rfx at sbus file dev/sbus/rfx.c rfx +device mgx: wsemuldisplaydev, rasops8, wsemul_sun +attach mgx at sbus +file dev/sbus/mgx.c mgx + device magma {} attach magma at sbus device mtty diff --git a/sys/dev/sbus/mgx.c b/sys/dev/sbus/mgx.c new file mode 100644 index 00000000000..fcedbd26f67 --- /dev/null +++ b/sys/dev/sbus/mgx.c @@ -0,0 +1,439 @@ +/* $OpenBSD: mgx.c,v 1.1 2004/06/20 21:28:06 miod Exp $ */ +/* + * Copyright (c) 2003, Miodrag Vallat. + * 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 ``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. + * + */ + +/* + * Driver for the Southlan Media Systems (now Quantum 3D) MGX and MGXPlus + * frame buffers. + * + * Pretty crude, due to the lack of documentation. Works as a dumb frame + * buffer in 8 bit mode, although the hardware can run in an 32 bit + * accelerated mode. Also, interrupts are not handled. + */ + +#include <sys/param.h> +#include <sys/systm.h> +#include <sys/buf.h> +#include <sys/device.h> +#include <sys/ioctl.h> +#include <sys/malloc.h> +#include <sys/mman.h> +#include <sys/tty.h> +#include <sys/conf.h> + +#include <uvm/uvm_extern.h> + +#include <machine/autoconf.h> +#include <machine/pmap.h> +#include <machine/cpu.h> +#include <machine/conf.h> + +#include <dev/wscons/wsconsio.h> +#include <dev/wscons/wsdisplayvar.h> +#include <dev/wscons/wscons_raster.h> +#include <dev/rasops/rasops.h> +#include <machine/fbvar.h> + +#include <dev/sbus/sbusvar.h> + +/* + * MGX PROM register layout + */ + +#define MGX_NREG 9 +#define MGX_REG_CRTC 4 /* video control and ramdac */ +#define MGX_REG_CTRL 5 /* control engine */ +#define MGX_REG_VRAM8 8 /* 8-bit memory space */ + +/* + * MGX CRTC empirical constants + */ +#define ADDRESS_REVERSE(x) ((x) ^ 0x03) +#define CRTC_COMMAND ADDRESS_REVERSE(0x03c4) +#define CRTC_DATA ADDRESS_REVERSE(0x03c5) +#define CRTC_CMAP ADDRESS_REVERSE(0x03c9) +#define CD_DISABLEVIDEO 0x0020 + +/* per-display variables */ +struct mgx_softc { + struct sunfb sc_sunfb; /* common base device */ + struct sbusdev sc_sd; /* sbus device */ + + bus_space_tag_t sc_bustag; + bus_addr_t sc_paddr; + + u_int8_t sc_cmap[256 * 3]; /* shadow colormap */ + volatile u_int8_t *sc_vidc; /* ramdac registers */ + + int sc_nscreens; +}; + +struct wsscreen_descr mgx_stdscreen = { + "std", +}; + +const struct wsscreen_descr *mgx_scrlist[] = { + &mgx_stdscreen, +}; + +struct wsscreen_list mgx_screenlist = { + sizeof(mgx_scrlist) / sizeof(struct wsscreen_descr *), + mgx_scrlist +}; + +int mgx_ioctl(void *, u_long, caddr_t, int, struct proc *); +int mgx_alloc_screen(void *, const struct wsscreen_descr *, void **, + int *, int *, long *); +void mgx_free_screen(void *, void *); +int mgx_show_screen(void *, void *, int, void (*cb)(void *, int, int), + void *); +paddr_t mgx_mmap(void *, off_t, int); +void mgx_setcolor(void *, u_int, u_int8_t, u_int8_t, u_int8_t); +int mgx_getcmap(u_int8_t *, struct wsdisplay_cmap *); +int mgx_putcmap(u_int8_t *, struct wsdisplay_cmap *); +void mgx_loadcmap(struct mgx_softc *, int, int); +void mgx_burner(void *, u_int ,u_int); + +struct wsdisplay_accessops mgx_accessops = { + mgx_ioctl, + mgx_mmap, + mgx_alloc_screen, + mgx_free_screen, + mgx_show_screen, + NULL, /* load_font */ + NULL, /* scrollback */ + NULL, /* getchar */ + mgx_burner +}; + +int mgxmatch(struct device *, void *, void *); +void mgxattach(struct device *, struct device *, void *); + +struct cfattach mgx_ca = { + sizeof(struct mgx_softc), mgxmatch, mgxattach +}; + +struct cfdriver mgx_cd = { + NULL, "mgx", DV_DULL +}; + +/* + * Match an MGX or MGX+ card. + */ +int +mgxmatch(struct device *parent, void *vcf, void *aux) +{ + struct sbus_attach_args *sa = aux; + + if (strcmp(sa->sa_name, "SMSI,mgx") != 0 && + strcmp(sa->sa_name, "mgx") != 0) + return (0); + + return (1); +} + +/* + * Attach an MGX frame buffer. + * This will keep the frame buffer in the actual PROM mode, and attach + * a wsdisplay child device to itself. + */ +void +mgxattach(struct device *parent, struct device *self, void *args) +{ + struct mgx_softc *sc = (struct mgx_softc *)self; + struct sbus_attach_args *sa = args; + struct wsemuldisplaydev_attach_args waa; + bus_space_tag_t bt; + bus_space_handle_t bh; + int node, fbsize; + int isconsole; + + bt = sa->sa_bustag; + node = sa->sa_node; + + printf(": %s", getpropstring(node, "model")); + + isconsole = node == fbnode; + + /* Check registers */ + if (sa->sa_nreg < MGX_NREG) { + printf("\n%s: expected %d registers, got %d\n", + self->dv_xname, MGX_NREG, sa->sa_nreg); + return; + } + + sc->sc_bustag = bt; + if (sbus_bus_map(bt, sa->sa_reg[MGX_REG_CRTC].sbr_slot, + sa->sa_reg[MGX_REG_CRTC].sbr_offset, PAGE_SIZE, + BUS_SPACE_MAP_LINEAR, 0, &bh) != 0) { + printf("\n%s: couldn't map crtc registers\n", self->dv_xname); + return; + } + sc->sc_vidc = (volatile u_int8_t *)bus_space_vaddr(bt, bh); + + /* enable video */ + mgx_burner(sc, 1, 0); + + fb_setsize(&sc->sc_sunfb, 8, 1152, 900, node, 0); + + /* Sanity check frame buffer memory */ + fbsize = getpropint(node, "fb_size", 0); + if (fbsize != 0 && sc->sc_sunfb.sf_fbsize > fbsize) { + printf("\n%s: expected at least %d bytes of vram, but card " + "only provides %d\n", + self->dv_xname, sc->sc_sunfb.sf_fbsize, fbsize); + return; + } + + /* Map the frame buffer memory area we're interested in */ + sc->sc_paddr = sbus_bus_addr(bt, sa->sa_reg[MGX_REG_VRAM8].sbr_slot, + sa->sa_reg[MGX_REG_VRAM8].sbr_offset); + if (sbus_bus_map(bt, sa->sa_reg[MGX_REG_VRAM8].sbr_slot, + sa->sa_reg[MGX_REG_VRAM8].sbr_offset, + round_page(sc->sc_sunfb.sf_fbsize), + BUS_SPACE_MAP_LINEAR, 0, &bh) != 0) { + printf("\n%s: couldn't map video memory\n", self->dv_xname); + return; + } + sc->sc_sunfb.sf_ro.ri_bits = bus_space_vaddr(bt, bh); + sc->sc_sunfb.sf_ro.ri_hw = sc; + + fbwscons_init(&sc->sc_sunfb, isconsole ? 0 : RI_CLEAR); + + bzero(sc->sc_cmap, sizeof(sc->sc_cmap)); + fbwscons_setcolormap(&sc->sc_sunfb, mgx_setcolor); + + mgx_stdscreen.capabilities = sc->sc_sunfb.sf_ro.ri_caps; + mgx_stdscreen.nrows = sc->sc_sunfb.sf_ro.ri_rows; + mgx_stdscreen.ncols = sc->sc_sunfb.sf_ro.ri_cols; + mgx_stdscreen.textops = &sc->sc_sunfb.sf_ro.ri_ops; + + printf(", %dx%d\n", + sc->sc_sunfb.sf_width, sc->sc_sunfb.sf_height); + + if (isconsole) { + fbwscons_console_init(&sc->sc_sunfb, + &mgx_stdscreen, -1, mgx_burner); + } + + sbus_establish(&sc->sc_sd, &sc->sc_sunfb.sf_dev); + + waa.console = isconsole; + waa.scrdata = &mgx_screenlist; + waa.accessops = &mgx_accessops; + waa.accesscookie = sc; + config_found(self, &waa, wsemuldisplaydevprint); +} + +/* + * wsdisplay operations + */ + +int +mgx_ioctl(void *dev, u_long cmd, caddr_t data, int flags, struct proc *p) +{ + struct mgx_softc *sc = dev; + struct wsdisplay_cmap *cm; + struct wsdisplay_fbinfo *wdf; + int error; + + switch (cmd) { + case WSDISPLAYIO_GTYPE: + *(u_int *)data = WSDISPLAY_TYPE_MGX; + break; + case WSDISPLAYIO_GINFO: + wdf = (struct wsdisplay_fbinfo *)data; + wdf->height = sc->sc_sunfb.sf_height; + wdf->width = sc->sc_sunfb.sf_width; + wdf->depth = sc->sc_sunfb.sf_depth; + wdf->cmsize = 256; + break; + case WSDISPLAYIO_LINEBYTES: + *(u_int *)data = sc->sc_sunfb.sf_linebytes; + break; + + case WSDISPLAYIO_GETCMAP: + cm = (struct wsdisplay_cmap *)data; + error = mgx_getcmap(sc->sc_cmap, cm); + if (error != 0) + return (error); + break; + case WSDISPLAYIO_PUTCMAP: + cm = (struct wsdisplay_cmap *)data; + error = mgx_putcmap(sc->sc_cmap, cm); + if (error != 0) + return (error); + mgx_loadcmap(sc, cm->index, cm->count); + break; + + default: + return (-1); + } + + return (0); +} + +paddr_t +mgx_mmap(void *v, off_t offset, int prot) +{ + struct mgx_softc *sc = v; + + if (offset & PGOFSET) + return (-1); + + /* Allow mapping as a dumb framebuffer from offset 0 */ + if (offset >= 0 && offset < sc->sc_sunfb.sf_fbsize) { + return (bus_space_mmap(sc->sc_bustag, sc->sc_paddr, + offset, prot, BUS_SPACE_MAP_LINEAR)); + } + + return (-1); +} + +int +mgx_alloc_screen(void *v, const struct wsscreen_descr *type, + void **cookiep, int *curxp, int *curyp, long *attrp) +{ + struct mgx_softc *sc = v; + + if (sc->sc_nscreens > 0) + return (ENOMEM); + + *cookiep = &sc->sc_sunfb.sf_ro; + *curyp = 0; + *curxp = 0; + sc->sc_sunfb.sf_ro.ri_ops.alloc_attr(&sc->sc_sunfb.sf_ro, + 0, 0, 0, attrp); + sc->sc_nscreens++; + return (0); +} + +void +mgx_free_screen(void *v, void *cookie) +{ + struct mgx_softc *sc = v; + + sc->sc_nscreens--; +} + +int +mgx_show_screen(void *v, void *cookie, int waitok, + void (*cb)(void *, int, int), void *cbarg) +{ + return (0); +} + +void +mgx_burner(void *v, u_int on, u_int flags) +{ + struct mgx_softc *sc = v; + + sc->sc_vidc[CRTC_COMMAND] = 1; /* trigger? */ + if (on) + sc->sc_vidc[CRTC_DATA] &= ~CD_DISABLEVIDEO; + else + sc->sc_vidc[CRTC_DATA] |= CD_DISABLEVIDEO; +} + +/* + * Colormap handling routines + */ + +void +mgx_setcolor(void *v, u_int index, u_int8_t r, u_int8_t g, u_int8_t b) +{ + struct mgx_softc *sc = v; + + index *= 3; + sc->sc_cmap[index++] = r; + sc->sc_cmap[index++] = g; + sc->sc_cmap[index] = b; + + mgx_loadcmap(sc, index, 1); +} + +void +mgx_loadcmap(struct mgx_softc *sc, int start, int ncolors) +{ + u_int8_t *color; + int i; + + /* + * Apparently there is no way to load an incomplete cmap to this + * DAC. What a waste. + */ + for (color = sc->sc_cmap, i = 0; i < 256 * 3; i++) + sc->sc_vidc[CRTC_CMAP] = *color++; +} + +int +mgx_getcmap(u_int8_t *cm, struct wsdisplay_cmap *rcm) +{ + u_int index = rcm->index, count = rcm->count, i; + int error; + + if (index >= 256 || count > 256 - index) + return (EINVAL); + + for (i = 0; i < count; i++) { + if ((error = + copyout(cm + (index + i) * 3 + 0, &rcm->red[i], 1)) != 0) + return (error); + if ((error = + copyout(cm + (index + i) * 3 + 1, &rcm->green[i], 1)) != 0) + return (error); + if ((error = + copyout(cm + (index + i) * 3 + 2, &rcm->blue[i], 1)) != 0) + return (error); + } + + return (0); +} + +int +mgx_putcmap(u_int8_t *cm, struct wsdisplay_cmap *rcm) +{ + u_int index = rcm->index, count = rcm->count, i; + int error; + + if (index >= 256 || count > 256 - index) + return (EINVAL); + + for (i = 0; i < count; i++) { + if ((error = + copyin(&rcm->red[i], cm + (index + i) * 3 + 0, 1)) != 0) + return (error); + if ((error = + copyin(&rcm->green[i], cm + (index + i) * 3 + 1, 1)) != 0) + return (error); + if ((error = + copyin(&rcm->blue[i], cm + (index + i) * 3 + 2, 1)) != 0) + return (error); + } + + return (0); +} |