summaryrefslogtreecommitdiff
path: root/sys/arch/sparc64/dev/gfxp.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/arch/sparc64/dev/gfxp.c')
-rw-r--r--sys/arch/sparc64/dev/gfxp.c108
1 files changed, 103 insertions, 5 deletions
diff --git a/sys/arch/sparc64/dev/gfxp.c b/sys/arch/sparc64/dev/gfxp.c
index 0cbdb23e5cb..88ee6b44448 100644
--- a/sys/arch/sparc64/dev/gfxp.c
+++ b/sys/arch/sparc64/dev/gfxp.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: gfxp.c,v 1.9 2009/06/28 12:36:33 kettenis Exp $ */
+/* $OpenBSD: gfxp.c,v 1.10 2009/06/28 13:40:51 kettenis Exp $ */
/*
* Copyright (c) 2009 Mark Kettenis.
@@ -59,6 +59,9 @@
#define PM2_OUT_FIFO 0x2000
#define PM2_SYNC_TAG 0x00000188
+#define PM2_PALETTE_WRITE_ADDR 0x4000
+#define PM2_PALETTE_DATA 0x4008
+
#define PM2V_INDEX_LOW 0x4020
#define PM2V_INDEX_HIGH 0x4028
#define PM2V_INDEX_DATA 0x4030
@@ -111,6 +114,9 @@ struct gfxp_softc {
pcitag_t sc_pcitag;
int sc_mode;
+ u_int8_t sc_cmap_red[256];
+ u_int8_t sc_cmap_green[256];
+ u_int8_t sc_cmap_blue[256];
/* Saved state to clean up after X11. */
uint32_t sc_read_mode;
@@ -145,6 +151,9 @@ struct cfdriver gfxp_cd = {
};
int gfxp_is_console(int);
+int gfxp_getcmap(struct gfxp_softc *, struct wsdisplay_cmap *);
+int gfxp_putcmap(struct gfxp_softc *, struct wsdisplay_cmap *);
+void gfxp_setcolor(void *, u_int, u_int8_t, u_int8_t, u_int8_t);
void gfxp_copycols(void *, int, int, int, int);
void gfxp_erasecols(void *, int, int, int, long);
@@ -234,6 +243,7 @@ gfxp_attach(struct device *parent, struct device *self, void *aux)
}
fbwscons_init(&sc->sc_sunfb, flags, console);
+ fbwscons_setcolormap(&sc->sc_sunfb, gfxp_setcolor);
sc->sc_mode = WSDISPLAYIO_MODE_EMUL;
gfxp_init(sc);
@@ -260,15 +270,22 @@ gfxp_ioctl(void *v, u_long cmd, caddr_t data, int flags, struct proc *p)
break;
case WSDISPLAYIO_SMODE:
sc->sc_mode = *(u_int *)data;
- if (sc->sc_mode == WSDISPLAYIO_MODE_EMUL)
+ if (sc->sc_mode == WSDISPLAYIO_MODE_EMUL) {
+ fbwscons_setcolormap(&sc->sc_sunfb, gfxp_setcolor);
+
+ /* Clean up the mess left behind by X. */
gfxp_reinit(sc);
+ }
break;
case WSDISPLAYIO_GINFO:
wdf = (void *)data;
wdf->height = sc->sc_sunfb.sf_height;
wdf->width = sc->sc_sunfb.sf_width;
wdf->depth = sc->sc_sunfb.sf_depth;
- wdf->cmsize = 0;
+ if (sc->sc_sunfb.sf_depth == 32)
+ wdf->cmsize = 0;
+ else
+ wdf->cmsize = 256;
break;
case WSDISPLAYIO_GETSUPPORTEDDEPTH:
if (sc->sc_sunfb.sf_depth == 32)
@@ -280,6 +297,11 @@ gfxp_ioctl(void *v, u_long cmd, caddr_t data, int flags, struct proc *p)
*(u_int *)data = sc->sc_sunfb.sf_linebytes;
break;
+ case WSDISPLAYIO_GETCMAP:
+ return gfxp_getcmap(sc, (struct wsdisplay_cmap *)data);
+ case WSDISPLAYIO_PUTCMAP:
+ return gfxp_putcmap(sc, (struct wsdisplay_cmap *)data);
+
case WSDISPLAYIO_GPCIID:
sel = (struct pcisel *)data;
sel->pc_bus = PCITAG_BUS(sc->sc_pcitag);
@@ -296,8 +318,6 @@ gfxp_ioctl(void *v, u_long cmd, caddr_t data, int flags, struct proc *p)
case WSDISPLAYIO_GCURMAX:
case WSDISPLAYIO_GCURSOR:
case WSDISPLAYIO_SCURSOR:
- case WSDISPLAYIO_GETCMAP:
- case WSDISPLAYIO_PUTCMAP:
default:
return -1; /* not supported yet */
}
@@ -354,6 +374,84 @@ gfxp_is_console(int node)
return (fbnode == node);
}
+int
+gfxp_getcmap(struct gfxp_softc *sc, struct wsdisplay_cmap *cm)
+{
+ u_int index = cm->index;
+ u_int count = cm->count;
+ int error;
+
+ if (index >= 256 || count > 256 - index)
+ return (EINVAL);
+
+ error = copyout(&sc->sc_cmap_red[index], cm->red, count);
+ if (error)
+ return (error);
+ error = copyout(&sc->sc_cmap_green[index], cm->green, count);
+ if (error)
+ return (error);
+ error = copyout(&sc->sc_cmap_blue[index], cm->blue, count);
+ if (error)
+ return (error);
+ return (0);
+}
+
+int
+gfxp_putcmap(struct gfxp_softc *sc, struct wsdisplay_cmap *cm)
+{
+ u_int index = cm->index;
+ u_int count = cm->count;
+ u_int i;
+ int error;
+ u_char *r, *g, *b;
+
+ if (index >= 256 || count > 256 - index)
+ return (EINVAL);
+
+ if ((error = copyin(cm->red, &sc->sc_cmap_red[index], count)) != 0)
+ return (error);
+ if ((error = copyin(cm->green, &sc->sc_cmap_green[index], count)) != 0)
+ return (error);
+ if ((error = copyin(cm->blue, &sc->sc_cmap_blue[index], count)) != 0)
+ return (error);
+
+ r = &sc->sc_cmap_red[index];
+ g = &sc->sc_cmap_green[index];
+ b = &sc->sc_cmap_blue[index];
+
+ gfxp_wait_fifo(sc, 1);
+ bus_space_write_4(sc->sc_mmiot, sc->sc_mmioh,
+ PM2_PALETTE_WRITE_ADDR, index);
+ for (i = 0; i < count; i++) {
+ gfxp_wait_fifo(sc, 3);
+ bus_space_write_4(sc->sc_mmiot, sc->sc_mmioh,
+ PM2_PALETTE_DATA, *r);
+ bus_space_write_4(sc->sc_mmiot, sc->sc_mmioh,
+ PM2_PALETTE_DATA, *g);
+ bus_space_write_4(sc->sc_mmiot, sc->sc_mmioh,
+ PM2_PALETTE_DATA, *b);
+ r++, g++, b++;
+ }
+ return (0);
+}
+
+void
+gfxp_setcolor(void *v, u_int index, u_int8_t r, u_int8_t g, u_int8_t b)
+{
+ struct gfxp_softc *sc = v;
+
+ sc->sc_cmap_red[index] = r;
+ sc->sc_cmap_green[index] = g;
+ sc->sc_cmap_blue[index] = b;
+
+ gfxp_wait_fifo(sc, 4);
+ bus_space_write_4(sc->sc_mmiot, sc->sc_mmioh,
+ PM2_PALETTE_WRITE_ADDR, index);
+ bus_space_write_4(sc->sc_mmiot, sc->sc_mmioh, PM2_PALETTE_DATA, r);
+ bus_space_write_4(sc->sc_mmiot, sc->sc_mmioh, PM2_PALETTE_DATA, g);
+ bus_space_write_4(sc->sc_mmiot, sc->sc_mmioh, PM2_PALETTE_DATA, b);
+}
+
/*
* Accelerated routines.
*/