summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMiod Vallat <miod@cvs.openbsd.org>2004-06-20 21:28:07 +0000
committerMiod Vallat <miod@cvs.openbsd.org>2004-06-20 21:28:07 +0000
commitd2027a358165d1a616aae1c4a7137b94ff0c7ac6 (patch)
tree736b54f22a6750e7397e0d01e9075655a36c156c
parentc967fc5433b4ef6dba0cc4729567fefc1f83b64b (diff)
Port mgx(4) to sparc64.
-rw-r--r--sys/arch/sparc64/conf/GENERIC5
-rw-r--r--sys/dev/sbus/files.sbus6
-rw-r--r--sys/dev/sbus/mgx.c439
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);
+}