diff options
author | Miod Vallat <miod@cvs.openbsd.org> | 2003-06-17 21:21:34 +0000 |
---|---|---|
committer | Miod Vallat <miod@cvs.openbsd.org> | 2003-06-17 21:21:34 +0000 |
commit | 226756de18f3297a52634613f2bd4e835b91ed1d (patch) | |
tree | e434188eb8fb928e3b35ad0337d0e1f72ccc2081 /sys/arch/sparc/dev | |
parent | c0549bc55fa12827d6b81b623d3e92f8ac575c32 (diff) |
New driver and wsdisplay type for the Weitek Power9000 and Brooktree Bt445
combination, found on the Tadpole SPARCbook 3 only (later model use the
Power9100 and are already supported).
Untested for lack of hardware and volunteers, but carefully crafted; if it
does not work out of the box, this should be a good starting point.
"worth having in the tree" millert@
Diffstat (limited to 'sys/arch/sparc/dev')
-rw-r--r-- | sys/arch/sparc/dev/bt445.c | 98 | ||||
-rw-r--r-- | sys/arch/sparc/dev/bt445reg.h | 53 | ||||
-rw-r--r-- | sys/arch/sparc/dev/bt445var.h | 31 | ||||
-rw-r--r-- | sys/arch/sparc/dev/p9000.c | 746 | ||||
-rw-r--r-- | sys/arch/sparc/dev/p9100.c | 433 |
5 files changed, 1074 insertions, 287 deletions
diff --git a/sys/arch/sparc/dev/bt445.c b/sys/arch/sparc/dev/bt445.c new file mode 100644 index 00000000000..33859e2c9d6 --- /dev/null +++ b/sys/arch/sparc/dev/bt445.c @@ -0,0 +1,98 @@ +/* $OpenBSD: bt445.c,v 1.1 2003/06/17 21:21:31 miod Exp $ */ +/* + * Copyright (c) 2003, Miodrag Vallat. + * + * 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. + */ + +/* + * A skeleton driver for the Brooktree BT445 ``Chameleon'' RAMDAC found + * on the early Tadpole SPARCbook 3 machines. + * + * It will not perform anything by itself, but frame buffer drivers can use + * its softc structure to be able to access the RAMDAC themselves - creating + * a ramdac API for just this specific case of separate attachments is not + * worth doing. + */ + +#include <sys/param.h> +#include <sys/systm.h> +#include <sys/device.h> +#include <sys/conf.h> + +#include <machine/autoconf.h> + +#include <sparc/dev/bt445reg.h> +#include <sparc/dev/bt445var.h> + +void btcham_attach(struct device *, struct device *, void *); +int btcham_match(struct device *, void *, void *); + +struct cfattach btcham_ca = { + sizeof(struct bt445_softc), btcham_match, btcham_attach +}; + +struct cfdriver btcham_cd = { + NULL, "btcham", DV_DULL +}; + +/* + * Match a chameleon + */ +int +btcham_match(struct device *parent, void *vcf, void *aux) +{ + struct confargs *ca = aux; + struct romaux *ra = &ca->ca_ra; + + if (strcmp("bt445", ra->ra_name) != 0) + return (0); + + return (1); +} + +/* + * Attach a chameleon + */ +void +btcham_attach(struct device *parent, struct device *self, void *args) +{ + struct confargs *ca = args; + struct bt445_softc *sc = (struct bt445_softc *)self; + int id, rev; + + /* + * Map the registers + */ + sc->sc_regs = mapiodev(ca->ca_ra.ra_reg, 0, ca->ca_ra.ra_len); + + /* + * Report a few values to please the user. + */ + sc->sc_regs[BT445_ADDRESS] = BT445_REGISTER_OFFSET(BT445_ID); + id = sc->sc_regs[BT445_REGISTER_INDEX(BT445_ID)]; + + sc->sc_regs[BT445_ADDRESS] = BT445_REGISTER_OFFSET(BT445_ID); + rev = sc->sc_regs[BT445_REGISTER_INDEX(BT445_ID)]; + + printf(": id 0x%x, revision 0x%x\n", id, rev); +} diff --git a/sys/arch/sparc/dev/bt445reg.h b/sys/arch/sparc/dev/bt445reg.h new file mode 100644 index 00000000000..ed6b4079c8f --- /dev/null +++ b/sys/arch/sparc/dev/bt445reg.h @@ -0,0 +1,53 @@ +/* $OpenBSD: bt445reg.h,v 1.1 2003/06/17 21:21:31 miod Exp $ */ +/* + * Copyright (c) 2003, Miodrag Vallat. + * + * 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. + */ + +/* + * Actual BT445 register layout + */ + +/* Read/write address */ +#define BT445_ADDRESS 0 + +/* + * Access to a register is done by programming the address register with + * the low 8 bits, and then reading or writing at the register pointed out + * by the high 8 bits. + */ +#define BT445_REGISTER_OFFSET(x) ((x) & 0xff) +#define BT445_REGISTER_INDEX(x) (((x) >> 8) & 0xff) + +/* Palette data - 3 r/w cycles par address, and autoincrement */ +#define BT445_PALDATA 1 + +/* Overlay palette data - 3 r/w cycles par address, and autoincrement */ +#define BT445_OVPALDATA 3 + +/* + * Various registers (very incomplete...) + */ +#define BT445_ID 0x0200 +#define BT445_REVISION 0x0201 + diff --git a/sys/arch/sparc/dev/bt445var.h b/sys/arch/sparc/dev/bt445var.h new file mode 100644 index 00000000000..956587c9956 --- /dev/null +++ b/sys/arch/sparc/dev/bt445var.h @@ -0,0 +1,31 @@ +/* $OpenBSD: bt445var.h,v 1.1 2003/06/17 21:21:31 miod Exp $ */ +/* + * Copyright (c) 2003, Miodrag Vallat. + * + * 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. + */ + +struct bt445_softc { + struct device sc_dev; + + volatile u_int8_t *sc_regs; /* RAMDAC registers */ +}; diff --git a/sys/arch/sparc/dev/p9000.c b/sys/arch/sparc/dev/p9000.c new file mode 100644 index 00000000000..50cd26d3227 --- /dev/null +++ b/sys/arch/sparc/dev/p9000.c @@ -0,0 +1,746 @@ +/* $OpenBSD: p9000.c,v 1.1 2003/06/17 21:21:32 miod Exp $ */ +/* + * Copyright (c) 2003, Miodrag Vallat. + * + * 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 Tadpole SPARCbook 3 on-board display. + * Heavily based on the p9100 driver. + */ + +#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 <sparc/dev/btreg.h> +#include <sparc/dev/btvar.h> +#include <sparc/dev/bt445reg.h> +#include <sparc/dev/bt445var.h> +#include <sparc/dev/sbusvar.h> + +#include <dev/ic/p9000.h> + +#include "tctrl.h" +#if NTCTRL > 0 +#include <sparc/dev/tctrlvar.h> +#endif + +/* per-display variables */ +struct p9000_softc { + struct sunfb sc_sunfb; /* common base part */ + struct sbusdev sc_sd; /* sbus device */ + struct rom_reg sc_phys; /* phys address description */ + volatile u_int8_t *sc_cmd; /* command registers (dac, etc) */ + volatile u_int8_t *sc_ctl; /* control registers (draw engine) */ + union bt_cmap sc_cmap; /* Brooktree color map */ + volatile u_int8_t *sc_ramdac; /* BT445 registers */ + struct intrhand sc_ih; + int sc_nscreens; + u_int32_t sc_junk; /* throwaway value */ +}; + +struct wsscreen_descr p9000_stdscreen = { + "std", +}; + +const struct wsscreen_descr *p9000_scrlist[] = { + &p9000_stdscreen, +}; + +struct wsscreen_list p9000_screenlist = { + sizeof(p9000_scrlist) / sizeof(struct wsscreen_descr *), + p9000_scrlist +}; + +int p9000_ioctl(void *, u_long, caddr_t, int, struct proc *); +int p9000_alloc_screen(void *, const struct wsscreen_descr *, void **, + int *, int *, long *); +void p9000_free_screen(void *, void *); +int p9000_show_screen(void *, void *, int, void (*cb)(void *, int, int), + void *); +paddr_t p9000_mmap(void *, off_t, int); +static __inline__ void p9000_loadcmap_deferred(struct p9000_softc *, + u_int, u_int); +void p9000_loadcmap_immediate(struct p9000_softc *, u_int, u_int); +void p9000_setcolor(void *, u_int, u_int8_t, u_int8_t, u_int8_t); +void p9000_burner(void *, u_int, u_int); +int p9000_intr(void *); + +struct wsdisplay_accessops p9000_accessops = { + p9000_ioctl, + p9000_mmap, + p9000_alloc_screen, + p9000_free_screen, + p9000_show_screen, + NULL, /* load_font */ + NULL, /* scrollback */ + NULL, /* getchar */ + p9000_burner, +}; + +void p9000_ras_init(struct p9000_softc *); +void p9000_ras_copycols(void *, int, int, int, int); +void p9000_ras_copyrows(void *, int, int, int); +void p9000_ras_do_cursor(struct rasops_info *); +void p9000_ras_erasecols(void *, int, int, int, long int); +void p9000_ras_eraserows(void *, int, int, long int); + +int p9000match(struct device *, void *, void *); +void p9000attach(struct device *, struct device *, void *); + +struct cfattach p9k_ca = { + sizeof (struct p9000_softc), p9000match, p9000attach +}; + +struct cfdriver p9k_cd = { + NULL, "p9k", DV_DULL +}; + +/* + * SBus registers mappings + */ +#define P9000_NREG 5 /* actually, 7 total */ +#define P9000_REG_CTL 0 +#define P9000_REG_CMD 1 +#define P9000_REG_VRAM 4 + +/* + * P9000 read/write macros + */ + +#define P9000_READ_CTL(sc,reg) \ + *(volatile u_int32_t *)((sc)->sc_ctl + (reg)) +#define P9000_READ_CMD(sc,reg) \ + *(volatile u_int32_t *)((sc)->sc_cmd + (reg)) + +#define P9000_WRITE_CTL(sc,reg,value) \ + *(volatile u_int32_t *)((sc)->sc_ctl + (reg)) = (value) +#define P9000_WRITE_CMD(sc,reg,value) \ + *(volatile u_int32_t *)((sc)->sc_cmd + (reg)) = (value) + +/* + * On the Tadpole, the first write to a register group is ignored until + * the proper group address is latched, which can be done by reading from the + * register group first. + * + * Register groups are 0x80 bytes long (i.e. it is necessary to force a read + * when writing to an adress which upper 25 bit differ from the previous + * read or write operation). + * + * This is specific to the Tadpole design, and not a limitation of the + * Power 9000 hardware. + */ +#define P9000_SELECT_SCR(sc) \ + (sc)->sc_junk = P9000_READ_CTL(sc, P9000_SYSTEM_CONFIG) +#define P9000_SELECT_VCR(sc) \ + (sc)->sc_junk = P9000_READ_CTL(sc, P9000_HCR) +#define P9000_SELECT_VRAM(sc) \ + (sc)->sc_junk = P9000_READ_CTL(sc, P9000_MCR) +#define P9000_SELECT_PE(sc) \ + (sc)->sc_junk = P9000_READ_CMD(sc, P9000_PE_STATUS) +#define P9000_SELECT_DE_LOW(sc) \ + (sc)->sc_junk = P9000_READ_CMD(sc, P9000_DE_FG_COLOR) +#define P9000_SELECT_DE_HIGH(sc) \ + (sc)->sc_junk = P9000_READ_CMD(sc, P9000_DE_PATTERN(0)) +#define P9000_SELECT_COORD(sc,field) \ + (sc)->sc_junk = P9000_READ_CMD(sc, field) + + +int +p9000match(struct device *parent, void *vcf, void *aux) +{ + struct confargs *ca = aux; + struct romaux *ra = &ca->ca_ra; + + if (strcmp("p9000", ra->ra_name)) + return (0); + + return (1); +} + +void +p9000attach(struct device *parent, struct device *self, void *args) +{ + struct p9000_softc *sc = (struct p9000_softc *)self; + struct confargs *ca = args; + struct wsemuldisplaydev_attach_args waa; + int node, row, isconsole, scr; + struct device *btdev; + extern struct cfdriver btcham_cd; + +#ifdef DIAGNOSTIC + if (ca->ca_ra.ra_nreg < P9000_NREG) { + printf(": expected %d registers, got only %d\n", + P9000_NREG, ca->ca_ra.ra_nreg); + return; + } +#endif + + /* + * Find the RAMDAC device. It should have attached before, since it + * attaches at obio. If, for some reason, it did not, it's not worth + * going any further. + * + * We rely upon the PROM to properly initialize the RAMDAC in a safe + * mode. + */ + btdev = btcham_cd.cd_ndevs != 0 ? btcham_cd.cd_devs[0] : NULL; + if (btdev != NULL) + sc->sc_ramdac = ((struct bt445_softc *)btdev)->sc_regs; + + if (sc->sc_ramdac == NULL) { + printf(": bt445 did not attach previously\n"); + return; + } + + sc->sc_phys = ca->ca_ra.ra_reg[P9000_REG_VRAM]; + + sc->sc_ctl = mapiodev(&(ca->ca_ra.ra_reg[P9000_REG_CTL]), 0, + ca->ca_ra.ra_reg[0].rr_len); + sc->sc_cmd = mapiodev(&(ca->ca_ra.ra_reg[P9000_REG_CMD]), 0, + ca->ca_ra.ra_reg[1].rr_len); + + node = ca->ca_ra.ra_node; + isconsole = node == fbnode; + + fb_setsize(&sc->sc_sunfb, 8, 640, 480, node, ca->ca_bustype); + sc->sc_sunfb.sf_ro.ri_bits = mapiodev(&sc->sc_phys, 0, + round_page(sc->sc_sunfb.sf_fbsize)); + sc->sc_sunfb.sf_ro.ri_hw = sc; + + P9000_SELECT_SCR(sc); + scr = P9000_READ_CTL(sc, P9000_SYSTEM_CONFIG); + + printf(": rev %x, %dx%d\n", scr & SCR_ID_MASK, + sc->sc_sunfb.sf_width, sc->sc_sunfb.sf_height); + + sc->sc_ih.ih_fun = p9000_intr; + sc->sc_ih.ih_arg = sc; + intr_establish(ca->ca_ra.ra_intr[0].int_pri, &sc->sc_ih, IPL_FB); + + /* Disable frame buffer interrupts */ + P9000_SELECT_SCR(sc); + P9000_WRITE_CTL(sc, P9000_INTERRUPT_ENABLE, IER_MASTER_ENABLE | 0); + + /* + * If the framebuffer width is under 1024x768, we will switch from the + * PROM font to the more adequate 8x16 font here. + * However, we need to adjust two things in this case: + * - the display row should be overrided from the current PROM metrics, + * to prevent us from overwriting the last few lines of text. + * - if the 80x34 screen would make a large margin appear around it, + * choose to clear the screen rather than keeping old prom output in + * the margins. + * XXX there should be a rasops "clear margins" feature + */ + fbwscons_init(&sc->sc_sunfb, + isconsole && (sc->sc_sunfb.sf_width >= 1024)); + fbwscons_setcolormap(&sc->sc_sunfb, p9000_setcolor); + + /* + * Plug-in accelerated console operations. + */ + sc->sc_sunfb.sf_ro.ri_ops.copycols = p9000_ras_copycols; + sc->sc_sunfb.sf_ro.ri_ops.copyrows = p9000_ras_copyrows; + sc->sc_sunfb.sf_ro.ri_ops.erasecols = p9000_ras_erasecols; + sc->sc_sunfb.sf_ro.ri_ops.eraserows = p9000_ras_eraserows; + sc->sc_sunfb.sf_ro.ri_do_cursor = p9000_ras_do_cursor; + p9000_ras_init(sc); + + p9000_stdscreen.capabilities = sc->sc_sunfb.sf_ro.ri_caps; + p9000_stdscreen.nrows = sc->sc_sunfb.sf_ro.ri_rows; + p9000_stdscreen.ncols = sc->sc_sunfb.sf_ro.ri_cols; + p9000_stdscreen.textops = &sc->sc_sunfb.sf_ro.ri_ops; + + sbus_establish(&sc->sc_sd, &sc->sc_sunfb.sf_dev); + + /* enable video */ + p9000_burner(sc, 1, 0); + + if (isconsole) { + if (sc->sc_sunfb.sf_width < 1024) + row = 0; /* screen has been cleared above */ + else + row = -1; + + fbwscons_console_init(&sc->sc_sunfb, &p9000_stdscreen, row, + p9000_burner); + } + + waa.console = isconsole; + waa.scrdata = &p9000_screenlist; + waa.accessops = &p9000_accessops; + waa.accesscookie = sc; + config_found(self, &waa, wsemuldisplaydevprint); +} + +int +p9000_ioctl(void *v, u_long cmd, caddr_t data, int flags, struct proc *p) +{ + struct p9000_softc *sc = v; + struct wsdisplay_fbinfo *wdf; + struct wsdisplay_cmap *cm; +#if NTCTRL > 0 + struct wsdisplay_param *dp; +#endif + int error; + + switch (cmd) { + + case WSDISPLAYIO_GTYPE: + *(u_int *)data = WSDISPLAY_TYPE_SB_P9000; + break; + + case WSDISPLAYIO_SMODE: + /* Restore proper acceleration state upon leaving X11 */ + if (*(u_int *)data == WSDISPLAYIO_MODE_EMUL) { + p9000_ras_init(sc); + } + 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 = bt_getcmap(&sc->sc_cmap, cm); + if (error) + return (error); + break; + + case WSDISPLAYIO_PUTCMAP: + cm = (struct wsdisplay_cmap *)data; + error = bt_putcmap(&sc->sc_cmap, cm); + if (error) + return (error); + p9000_loadcmap_deferred(sc, cm->index, cm->count); + break; + +#if NTCTRL > 0 + case WSDISPLAYIO_GETPARAM: + dp = (struct wsdisplay_param *)data; + + switch (dp->param) { + case WSDISPLAYIO_PARAM_BRIGHTNESS: + dp->min = 0; + dp->max = 255; + dp->curval = tadpole_get_brightness(); + break; + case WSDISPLAYIO_PARAM_BACKLIGHT: + dp->min = 0; + dp->max = 1; + dp->curval = tadpole_get_video(); + break; + default: + return (-1); + } + break; + + case WSDISPLAYIO_SETPARAM: + dp = (struct wsdisplay_param *)data; + + switch (dp->param) { + case WSDISPLAYIO_PARAM_BRIGHTNESS: + tadpole_set_brightness(dp->curval); + break; + case WSDISPLAYIO_PARAM_BACKLIGHT: + tadpole_set_video(dp->curval); + break; + default: + return (-1); + } + break; +#endif /* NTCTRL > 0 */ + + case WSDISPLAYIO_SVIDEO: + case WSDISPLAYIO_GVIDEO: + case WSDISPLAYIO_GCURPOS: + case WSDISPLAYIO_SCURPOS: + case WSDISPLAYIO_GCURMAX: + case WSDISPLAYIO_GCURSOR: + case WSDISPLAYIO_SCURSOR: + default: + return (-1); /* not supported yet */ + } + + return (0); +} + +int +p9000_alloc_screen(void *v, const struct wsscreen_descr *type, void **cookiep, + int *curxp, int *curyp, long *attrp) +{ + struct p9000_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, + WSCOL_BLACK, WSCOL_WHITE, WSATTR_WSCOLORS, attrp); + sc->sc_nscreens++; + return (0); +} + +void +p9000_free_screen(void *v, void *cookie) +{ + struct p9000_softc *sc = v; + + sc->sc_nscreens--; +} + +int +p9000_show_screen(void *v, void *cookie, int waitok, + void (*cb)(void *, int, int), void *cbarg) +{ + return (0); +} + +/* + * Return the address that would map the given device at the given + * offset, allowing for the given protection, or return -1 for error. + */ +paddr_t +p9000_mmap(void *v, off_t offset, int prot) +{ + struct p9000_softc *sc = v; + + if (offset & PGOFSET) + return (-1); + + if (offset >= 0 && offset < sc->sc_sunfb.sf_fbsize) { + return (REG2PHYS(&sc->sc_phys, offset) | PMAP_NC); + } + + return (-1); +} + +void +p9000_setcolor(void *v, u_int index, u_int8_t r, u_int8_t g, u_int8_t b) +{ + struct p9000_softc *sc = v; + union bt_cmap *bcm = &sc->sc_cmap; + + bcm->cm_map[index][0] = r; + bcm->cm_map[index][1] = g; + bcm->cm_map[index][2] = b; + p9000_loadcmap_immediate(sc, index, 1); +} + +void +p9000_loadcmap_immediate(struct p9000_softc *sc, u_int start, u_int ncolors) +{ + sc->sc_ramdac[BT445_ADDRESS] = start; + for (ncolors += start; start < ncolors; start++) { + sc->sc_ramdac[BT445_PALDATA] = sc->sc_cmap.cm_map[start][0]; + sc->sc_ramdac[BT445_PALDATA] = sc->sc_cmap.cm_map[start][1]; + sc->sc_ramdac[BT445_PALDATA] = sc->sc_cmap.cm_map[start][2]; + } +} + +static __inline__ void +p9000_loadcmap_deferred(struct p9000_softc *sc, u_int start, u_int ncolors) +{ + /* Schedule an interrupt for next retrace */ + P9000_SELECT_SCR(sc); + P9000_WRITE_CTL(sc, P9000_INTERRUPT_ENABLE, + IER_MASTER_ENABLE | IER_MASTER_INTERRUPT | + IER_VBLANK_ENABLE | IER_VBLANK_INTERRUPT); +} + +void +p9000_burner(void *v, u_int on, u_int flags) +{ + struct p9000_softc *sc = v; + u_int32_t vcr; + int s; + + s = splhigh(); + P9000_SELECT_VCR(sc); + vcr = P9000_READ_CTL(sc, P9000_SRTC1); + if (on) + vcr |= SRTC1_VIDEN; + else + vcr &= ~SRTC1_VIDEN; + P9000_WRITE_CTL(sc, P9000_SRTC1, vcr); +#if NTCTRL > 0 + tadpole_set_video(on); +#endif + splx(s); +} + +int +p9000_intr(void *v) +{ + struct p9000_softc *sc = v; + + if (P9000_READ_CTL(sc, P9000_INTERRUPT) & IER_VBLANK_INTERRUPT) { + p9000_loadcmap_immediate(sc, 0, 256); + + /* Disable further interrupts now */ + /* P9000_SELECT_SCR(sc); */ + P9000_WRITE_CTL(sc, P9000_INTERRUPT_ENABLE, + IER_MASTER_ENABLE | 0); + + return (1); + } + + return (0); +} + +/* + * Accelerated text console code + */ + +static __inline__ void p9000_drain(struct p9000_softc *); + +static __inline__ void +p9000_drain(struct p9000_softc *sc) +{ + while (P9000_READ_CMD(sc, P9000_PE_STATUS) & + (STATUS_QUAD_BUSY | STATUS_BLIT_BUSY)); +} + +void +p9000_ras_init(struct p9000_softc *sc) +{ + /* + * Setup safe defaults for the parameter and drawing engine, in + * order to minimize the operations to do for ri_ops. + */ + + P9000_SELECT_DE_LOW(sc); + P9000_WRITE_CMD(sc, P9000_DE_DRAWMODE, + DM_PICK_CONTROL | 0 | DM_BUFFER_CONTROL | DM_BUFFER_ENABLE0); + + P9000_WRITE_CMD(sc, P9000_DE_PATTERN_ORIGIN_X, 0); + P9000_WRITE_CMD(sc, P9000_DE_PATTERN_ORIGIN_Y, 0); + /* enable all planes */ + P9000_WRITE_CMD(sc, P9000_DE_PLANEMASK, 0xff); + + /* Unclip */ + P9000_WRITE_CMD(sc, P9000_DE_WINMIN, 0); + P9000_WRITE_CMD(sc, P9000_DE_WINMAX, + P9000_COORDS(sc->sc_sunfb.sf_width - 1, sc->sc_sunfb.sf_height - 1)); + + P9000_SELECT_PE(sc); + P9000_WRITE_CMD(sc, P9000_PE_WINOFFSET, 0); +} + +void +p9000_ras_copycols(void *v, int row, int src, int dst, int n) +{ + struct rasops_info *ri = v; + struct p9000_softc *sc = ri->ri_hw; + + n *= ri->ri_font->fontwidth; + n--; + src *= ri->ri_font->fontwidth; + src += ri->ri_xorigin; + dst *= ri->ri_font->fontwidth; + dst += ri->ri_xorigin; + row *= ri->ri_font->fontheight; + row += ri->ri_yorigin; + + p9000_drain(sc); + P9000_SELECT_DE_LOW(sc); + P9000_WRITE_CMD(sc, P9000_DE_RASTER, + P9000_RASTER_SRC & P9000_RASTER_MASK); + + P9000_SELECT_COORD(sc, P9000_DC_COORD(0)); + P9000_WRITE_CMD(sc, P9000_DC_COORD(0) + P9000_COORD_XY, + P9000_COORDS(src, row)); + P9000_WRITE_CMD(sc, P9000_DC_COORD(1) + P9000_COORD_XY, + P9000_COORDS(src + n, row + ri->ri_font->fontheight - 1)); + P9000_SELECT_COORD(sc, P9000_DC_COORD(2)); + P9000_WRITE_CMD(sc, P9000_DC_COORD(2) + P9000_COORD_XY, + P9000_COORDS(dst, row)); + P9000_WRITE_CMD(sc, P9000_DC_COORD(3) + P9000_COORD_XY, + P9000_COORDS(dst + n, row + ri->ri_font->fontheight - 1)); + + sc->sc_junk = P9000_READ_CMD(sc, P9000_PE_BLIT); + + p9000_drain(sc); +} + +void +p9000_ras_copyrows(void *v, int src, int dst, int n) +{ + struct rasops_info *ri = v; + struct p9000_softc *sc = ri->ri_hw; + + n *= ri->ri_font->fontheight; + n--; + src *= ri->ri_font->fontheight; + src += ri->ri_yorigin; + dst *= ri->ri_font->fontheight; + dst += ri->ri_yorigin; + + p9000_drain(sc); + P9000_SELECT_DE_LOW(sc); + P9000_WRITE_CMD(sc, P9000_DE_RASTER, + P9000_RASTER_SRC & P9000_RASTER_MASK); + + P9000_SELECT_COORD(sc, P9000_DC_COORD(0)); + P9000_WRITE_CMD(sc, P9000_DC_COORD(0) + P9000_COORD_XY, + P9000_COORDS(ri->ri_xorigin, src)); + P9000_WRITE_CMD(sc, P9000_DC_COORD(1) + P9000_COORD_XY, + P9000_COORDS(ri->ri_xorigin + ri->ri_emuwidth - 1, src + n)); + P9000_SELECT_COORD(sc, P9000_DC_COORD(2)); + P9000_WRITE_CMD(sc, P9000_DC_COORD(2) + P9000_COORD_XY, + P9000_COORDS(ri->ri_xorigin, dst)); + P9000_WRITE_CMD(sc, P9000_DC_COORD(3) + P9000_COORD_XY, + P9000_COORDS(ri->ri_xorigin + ri->ri_emuwidth - 1, dst + n)); + + sc->sc_junk = P9000_READ_CMD(sc, P9000_PE_BLIT); + + p9000_drain(sc); +} + +void +p9000_ras_erasecols(void *v, int row, int col, int n, long int attr) +{ + struct rasops_info *ri = v; + struct p9000_softc *sc = ri->ri_hw; + int fg, bg; + + rasops_unpack_attr(attr, &fg, &bg, NULL); + + n *= ri->ri_font->fontwidth; + col *= ri->ri_font->fontwidth; + col += ri->ri_xorigin; + row *= ri->ri_font->fontheight; + row += ri->ri_yorigin; + + p9000_drain(sc); + P9000_SELECT_DE_LOW(sc); + P9000_WRITE_CMD(sc, P9000_DE_RASTER, + P9000_RASTER_PATTERN & P9000_RASTER_MASK); + P9000_WRITE_CMD(sc, P9000_DE_FG_COLOR, bg); + + P9000_SELECT_COORD(sc, P9000_LC_RECT); + P9000_WRITE_CMD(sc, P9000_LC_RECT + P9000_COORD_XY, + P9000_COORDS(col, row)); + P9000_WRITE_CMD(sc, P9000_LC_RECT + P9000_COORD_XY, + P9000_COORDS(col + n, row + ri->ri_font->fontheight)); + + sc->sc_junk = P9000_READ_CMD(sc, P9000_PE_QUAD); + + p9000_drain(sc); +} + +void +p9000_ras_eraserows(void *v, int row, int n, long int attr) +{ + struct rasops_info *ri = v; + struct p9000_softc *sc = ri->ri_hw; + int fg, bg; + + rasops_unpack_attr(attr, &fg, &bg, NULL); + + p9000_drain(sc); + P9000_SELECT_DE_LOW(sc); + P9000_WRITE_CMD(sc, P9000_DE_RASTER, + P9000_RASTER_PATTERN & P9000_RASTER_MASK); + P9000_WRITE_CMD(sc, P9000_DE_FG_COLOR, bg); + + P9000_SELECT_COORD(sc, P9000_LC_RECT); + if (n == ri->ri_rows && ISSET(ri->ri_flg, RI_FULLCLEAR)) { + P9000_WRITE_CMD(sc, P9000_LC_RECT + P9000_COORD_XY, + P9000_COORDS(0, 0)); + P9000_WRITE_CMD(sc, P9000_LC_RECT + P9000_COORD_XY, + P9000_COORDS(ri->ri_width, ri->ri_height)); + } else { + n *= ri->ri_font->fontheight; + row *= ri->ri_font->fontheight; + row += ri->ri_yorigin; + + P9000_WRITE_CMD(sc, P9000_LC_RECT + P9000_COORD_XY, + P9000_COORDS(ri->ri_xorigin, row)); + P9000_WRITE_CMD(sc, P9000_LC_RECT + P9000_COORD_XY, + P9000_COORDS(ri->ri_xorigin + ri->ri_emuwidth, row + n)); + } + + sc->sc_junk = P9000_READ_CMD(sc, P9000_PE_QUAD); + + p9000_drain(sc); +} + +void +p9000_ras_do_cursor(struct rasops_info *ri) +{ + struct p9000_softc *sc = ri->ri_hw; + int row, col; + + row = ri->ri_crow * ri->ri_font->fontheight + ri->ri_yorigin; + col = ri->ri_ccol * ri->ri_font->fontwidth + ri->ri_xorigin; + + p9000_drain(sc); + + P9000_SELECT_DE_LOW(sc); + P9000_WRITE_CMD(sc, P9000_DE_RASTER, + (~P9000_RASTER_DST) & P9000_RASTER_MASK); + + P9000_SELECT_COORD(sc, P9000_LC_RECT); + P9000_WRITE_CMD(sc, P9000_LC_RECT + P9000_COORD_XY, + P9000_COORDS(col, row)); + P9000_WRITE_CMD(sc, P9000_LC_RECT + P9000_COORD_XY, + P9000_COORDS(col + ri->ri_font->fontwidth, + row + ri->ri_font->fontheight)); + + sc->sc_junk = P9000_READ_CMD(sc, P9000_PE_QUAD); + + p9000_drain(sc); +} diff --git a/sys/arch/sparc/dev/p9100.c b/sys/arch/sparc/dev/p9100.c index 543aa346f67..eeaf1ed590e 100644 --- a/sys/arch/sparc/dev/p9100.c +++ b/sys/arch/sparc/dev/p9100.c @@ -1,4 +1,4 @@ -/* $OpenBSD: p9100.c,v 1.22 2003/06/13 23:40:12 miod Exp $ */ +/* $OpenBSD: p9100.c,v 1.23 2003/06/17 21:21:32 miod Exp $ */ /* * Copyright (c) 2003, Miodrag Vallat. @@ -28,8 +28,9 @@ */ /* - * color display (p9100) driver. Based on cgthree.c and the NetBSD - * p9100 driver. + * color display (p9100) driver. + * Initially based on cgthree.c and the NetBSD p9100 driver, then hacked + * beyond recognition. */ #include <sys/param.h> @@ -59,6 +60,8 @@ #include <sparc/dev/btvar.h> #include <sparc/dev/sbusvar.h> +#include <dev/ic/p9000.h> + #include "tctrl.h" #if NTCTRL > 0 #include <sparc/dev/tctrlvar.h> @@ -69,8 +72,8 @@ struct p9100_softc { struct sunfb sc_sunfb; /* common base part */ struct sbusdev sc_sd; /* sbus device */ struct rom_reg sc_phys; /* phys address description */ - struct p9100_cmd *sc_cmd; /* command registers (dac, etc) */ - struct p9100_ctl *sc_ctl; /* control registers (draw engine) */ + volatile u_int8_t *sc_cmd; /* command registers (dac, etc) */ + volatile u_int8_t *sc_ctl; /* control registers (draw engine) */ union bt_cmap sc_cmap; /* Brooktree color map */ struct intrhand sc_ih; int sc_nscreens; @@ -143,206 +146,44 @@ struct cfdriver pnozz_cd = { #define P9100_REG_VRAM 2 /* - * System control and command registers - * (IBM RGB528 RamDac, p9100, video coprocessor) + * IBM RGB525 RAMDAC registers */ -struct p9100_ctl { - /* System control registers: 0x0000 - 0x00ff */ - struct p9100_scr { - volatile u_int32_t :32; - volatile u_int32_t scr; /* system config reg */ -#define SCR_ID_MASK 0x00000007 -#define SCR_PIXEL_MASK 0x1c000000 -#define SCR_PIXEL_8BPP 0x08000000 -#define SCR_PIXEL_16BPP 0x0c000000 -#define SCR_PIXEL_24BPP 0x1c000000 -#define SCR_PIXEL_32BPP 0x14000000 - volatile u_int32_t ir; /* interrupt reg */ - volatile u_int32_t ier; /* interrupt enable */ -#define IER_MASTER_ENABLE 0x00000080 -#define IER_MASTER_INTERRUPT 0x00000040 -#define IER_VBLANK_ENABLE 0x00000020 -#define IER_VBLANK_INTERRUPT 0x00000010 -#define IER_PICK_ENABLE 0x00000008 -#define IER_PICK_INTERRUPT 0x00000004 -#define IER_IDLE_ENABLE 0x00000002 -#define IER_IDLE_INTERRUPT 0x00000001 - volatile u_int32_t arbr; /* alt read bank reg */ - volatile u_int32_t awbr; /* alt write bank reg */ - volatile u_int32_t unused1[58]; - } ctl_scr; - - /* Video control registers: 0x0100 - 0x017f */ - struct p9100_vcr { - volatile u_int32_t :32; - volatile u_int32_t hcr; /* horizontal cntr */ - volatile u_int32_t htr; /* horizontal total */ - volatile u_int32_t hsre; /* horiz sync rising */ - volatile u_int32_t hbre; /* horiz blank rising */ - volatile u_int32_t hbfe; /* horiz blank fallng */ - volatile u_int32_t hcp; /* horiz cntr preload */ - volatile u_int32_t vcr; /* vertical cntr */ - volatile u_int32_t vl; /* vertical length */ - volatile u_int32_t vsre; /* vert sync rising */ - volatile u_int32_t vbre; /* vert blank rising */ - volatile u_int32_t vbfe; /* vert blank fallng */ - volatile u_int32_t vcp; /* vert cntr preload */ - volatile u_int32_t sra; /* scrn repaint addr */ - volatile u_int32_t srtc1; /* scrn rpnt time 1 */ -#define SRTC1_VIDEN 0x00000020 - volatile u_int32_t qsf; /* qsf counter */ - volatile u_int32_t srtc2; /* scrn rpnt time 2 */ - volatile u_int32_t unused1[15]; - } ctl_vcr; - - /* VRAM control registers: 0x0180 - 0x1ff */ - struct p9100_vram { - volatile u_int32_t :32; - volatile u_int32_t mc; /* memory config */ - volatile u_int32_t rp; /* refresh period */ - volatile u_int32_t rc; /* refresh count */ - volatile u_int32_t rasmax; /* ras low maximum */ - volatile u_int32_t rascur; /* ras low current */ - volatile u_int32_t dacfifo; /* free fifo */ - volatile u_int32_t unused1[25]; - } ctl_vram; - - /* IBM RGB528 RAMDAC registers: 0x0200 - 0x3ff */ - struct p9100_dac { - volatile u_int32_t pwraddr; /* wr palette address */ - volatile u_int32_t paldata; /* palette data */ - volatile u_int32_t pixmask; /* pixel mask */ - volatile u_int32_t prdaddr; /* rd palette address */ - volatile u_int32_t idxlow; /* reg index low */ - volatile u_int32_t idxhigh; /* reg index high */ - volatile u_int32_t regdata; /* register data */ - volatile u_int32_t idxctrl; /* index control */ - volatile u_int32_t unused1[120]; - } ctl_dac; - - /* Video coprocessor interface: 0x0400 - 0x1fff */ - volatile u_int32_t ctl_vci[768]; -}; + +/* Palette write address */ +#define IBM525_WRADDR 0 +/* Palette data */ +#define IBM525_DATA 1 +/* Pixel mask */ +#define IBM525_PIXMASK 2 +/* Read palette address */ +#define IBM525_RDADDR 3 +/* Register index low */ +#define IBM525_IDXLOW 4 +/* Register index high */ +#define IBM525_IDXHIGH 5 +/* Register data */ +#define IBM525_REGDATA 6 +/* Index control */ +#define IBM525_IDXCONTROL 7 /* - * Power 9100 Drawing engine + * P9100 read/write macros */ -struct p9100_coord { - volatile u_int32_t :32; - volatile u_int32_t :32; - volatile u_int32_t abs_x32; - volatile u_int32_t :32; - volatile u_int32_t abs_y32; - volatile u_int32_t :32; - volatile u_int32_t abs_x16y16; - volatile u_int32_t :32; - volatile u_int32_t :32; - volatile u_int32_t :32; - volatile u_int32_t rel_x32; - volatile u_int32_t :32; - volatile u_int32_t rel_y32; - volatile u_int32_t :32; - volatile u_int32_t rel_x16y16; - volatile u_int32_t :32; -}; - -/* How to encode a x16y16 value */ -#define P9100_COORDS(col,row) ((((col) & 0x0fff) << 16) | ((row) & 0x0fff)) - -struct p9100_cmd { - /* Parameter engine 0x2000-0x217f */ - struct cmd_pe { - volatile u_int32_t status; /* status register */ -#define STATUS_QUAD_BUSY 0x80000000 -#define STATUS_BLIT_BUSY 0x40000000 -#define STATUS_PICK_DETECTED 0x00000080 -#define STATUS_PIXEL_ERROR 0x00000040 -#define STATUS_BLIT_ERROR 0x00000020 -#define STATUS_QUAD_ERROR 0x00000010 -#define STATUS_QUAD_CONCAVE 0x00000008 -#define STATUS_QUAD_OUTSIDE 0x00000004 -#define STATUS_QUAD_INSIDE 0x00000002 -#define STATUS_QUAD_STRADDLE 0x00000001 - volatile u_int32_t blit; /* blit command */ - volatile u_int32_t quad; /* quad command */ - volatile u_int32_t pixel8; /* P9000 pixel8 command */ - volatile u_int32_t :32; - volatile u_int32_t nextpixels; /* next_pixels command */ - volatile u_int32_t unused1[(0x80 - 0x18) / 4]; - volatile u_int32_t pixel1[32]; /* pixel1 command */ - volatile u_int32_t unused2[(0x180 - 0x100) / 4]; - - /* Parameter engine registers 0x2180-0x21ff */ - volatile u_int32_t :32; - volatile u_int32_t oor; /* out of range register */ - volatile u_int32_t :32; - volatile u_int32_t cindex; /* index register */ - volatile u_int32_t winoffset; /* window offset (16+16) */ - volatile u_int32_t winmin; /* window minimum */ - volatile u_int32_t winmax; /* window maximum */ - volatile u_int32_t :32; - volatile u_int32_t yclip; - volatile u_int32_t xclip; - volatile u_int32_t xedgeless; - volatile u_int32_t xedgegreater; - volatile u_int32_t yedgeless; - volatile u_int32_t yedgegreater; - volatile u_int32_t unused3[(0x200 - 0x1b8) / 4]; - } pe; - - /* Drawing engine 0x2200-0x2fff */ - struct cmd_de { - volatile u_int32_t color0; /* fg color */ - volatile u_int32_t color1; /* bg color */ - volatile u_int32_t planemask; /* 8-bit plane mask */ - volatile u_int32_t drawmode; -#define DM_PICK_CONTROL 0x00000008 -#define DM_PICK_ENABLE 0x00000004 -#define DM_BUFFER_CONTROL 0x00000002 -#define DM_BUFFER_ENABLE0 0x00000000 -#define DM_BUFFER_ENABLE1 0x00000001 - volatile u_int32_t patternx; /* Quad 16x16 pattern origin */ - volatile u_int32_t patterny; /* (4-bit) */ - volatile u_int32_t raster; - volatile u_int32_t pixel8; /* Pixel8 extra storage */ - volatile u_int32_t winmin; /* window min, same as above */ - volatile u_int32_t winmax; /* window max, same as above */ - volatile u_int32_t unused1[(0x38 - 0x28) / 4]; - volatile u_int32_t color2; - volatile u_int32_t color3; - volatile u_int32_t unused2[(0x80 - 0x40) / 4]; - volatile u_int32_t pattern[4]; /* pattern for quad fill */ - volatile u_int32_t user[4]; /* user defined registers */ - volatile u_int32_t bwinmin; /* byte window minimum */ - volatile u_int32_t bwinmax; /* byte window maximum */ - volatile u_int32_t unused3[(0x3000 - 0x22a8) / 4]; - } de; - - /* Device coordinates 0x3000-0x31ff */ - struct cmd_dc { - struct p9100_coord coord[4]; - volatile u_int32_t unused[(0x200 - 0x100) / 4]; - } dc; - - /* Load coordinates 0x3200-0x33ff */ - struct cmd_lc { - struct p9100_coord point; - struct p9100_coord line; - struct p9100_coord tri; - struct p9100_coord quad; - struct p9100_coord rect; - } lc; -}; - -/* How to encode a color in 8 bit mode, for the drawing engine */ -#define P9100_COLOR8(col) ((col) | ((col) << 8) | ((col) << 16)) - -/* P9100 minterms for the raster register */ -#define P9100_RASTER_SRC 0x00cc -#define P9100_RASTER_DST 0x00aa -#define P9100_RASTER_PATTERN 0x00f0 -#define P9100_RASTER_MASK 0x00ff +#define P9100_READ_CTL(sc,reg) \ + *(volatile u_int32_t *)((sc)->sc_ctl + (reg)) +#define P9100_READ_CMD(sc,reg) \ + *(volatile u_int32_t *)((sc)->sc_cmd + (reg)) +#define P9100_READ_RAMDAC(sc,reg) \ + *(volatile u_int32_t *)((sc)->sc_ctl + P9100_RAMDAC_REGISTER(reg)) + +#define P9100_WRITE_CTL(sc,reg,value) \ + *(volatile u_int32_t *)((sc)->sc_ctl + (reg)) = (value) +#define P9100_WRITE_CMD(sc,reg,value) \ + *(volatile u_int32_t *)((sc)->sc_cmd + (reg)) = (value) +#define P9100_WRITE_RAMDAC(sc,reg,value) \ + *(volatile u_int32_t *)((sc)->sc_ctl + P9100_RAMDAC_REGISTER(reg)) = \ + (value) /* * On the Tadpole, the first write to a register group is ignored until @@ -356,17 +197,22 @@ struct p9100_cmd { * This is specific to the Tadpole design, and not a limitation of the * Power 9100 hardware. */ -#define P9100_SELECT_SCR(sc) ((sc)->sc_junk = (sc)->sc_ctl->ctl_scr.scr) -#define P9100_SELECT_VCR(sc) ((sc)->sc_junk = (sc)->sc_ctl->ctl_vcr.hcr) -#define P9100_SELECT_VRAM(sc) ((sc)->sc_junk = (sc)->sc_ctl->ctl_vram.mc) -#define P9100_SELECT_DAC(sc) ((sc)->sc_junk = (sc)->sc_ctl->ctl_dac.pwraddr) -#define P9100_SELECT_VCI(sc) ((sc)->sc_junk = (sc)->sc_ctl->ctl_vci[0]) -#define P9100_SELECT_DE_LOW(sc) ((sc)->sc_junk = (sc)->sc_cmd->de.color0) +#define P9100_SELECT_SCR(sc) \ + (sc)->sc_junk = P9100_READ_CTL(sc, P9000_SYSTEM_CONFIG) +#define P9100_SELECT_VCR(sc) \ + (sc)->sc_junk = P9100_READ_CTL(sc, P9000_HCR) +#define P9100_SELECT_VRAM(sc) \ + (sc)->sc_junk = P9100_READ_CTL(sc, P9000_MCR) +#define P9100_SELECT_DAC(sc) \ + (sc)->sc_junk = P9100_READ_CTL(sc, P9100_RAMDAC_REGISTER(0)) +#define P9100_SELECT_PE(sc) \ + (sc)->sc_junk = P9100_READ_CMD(sc, P9000_PE_STATUS) +#define P9100_SELECT_DE_LOW(sc) \ + (sc)->sc_junk = P9100_READ_CMD(sc, P9000_DE_FG_COLOR) #define P9100_SELECT_DE_HIGH(sc) \ - ((sc)->sc_junk = (sc)->sc_cmd->de.bwinmax) -#define P9100_SELECT_PE(sc) ((sc)->sc_junk = (sc)->sc_cmd->pe.cindex) + (sc)->sc_junk = P9100_READ_CMD(sc, P9000_DE_PATTERN(0)) #define P9100_SELECT_COORD(sc,field) \ - ((sc)->sc_junk = (sc)->sc_cmd->##field##.abs_x32) + (sc)->sc_junk = P9100_READ_CMD(sc, field) /* * For some reason, every write to a DAC register needs to be followed by a @@ -376,7 +222,7 @@ struct p9100_cmd { #define P9100_FLUSH_DAC(sc) \ do { \ P9100_SELECT_VRAM(sc); \ - (sc)->sc_junk = (sc)->sc_ctl->ctl_vram.dacfifo; \ + (sc)->sc_junk = P9100_READ_CTL(sc, P9100_FREE_FIFO); \ } while (0) int @@ -422,7 +268,7 @@ p9100attach(struct device *parent, struct device *self, void *args) isconsole = node == fbnode; P9100_SELECT_SCR(sc); - scr = sc->sc_ctl->ctl_scr.scr; + scr = P9100_READ_CTL(sc, P9000_SYSTEM_CONFIG); switch (scr & SCR_PIXEL_MASK) { case SCR_PIXEL_32BPP: fb_depth = 32; @@ -457,7 +303,7 @@ p9100attach(struct device *parent, struct device *self, void *args) /* Disable frame buffer interrupts */ P9100_SELECT_SCR(sc); - sc->sc_ctl->ctl_scr.ier = IER_MASTER_ENABLE | 0; + P9100_WRITE_CTL(sc, P9000_INTERRUPT_ENABLE, IER_MASTER_ENABLE | 0); /* * If the framebuffer width is under 1024x768, we will switch from the @@ -471,7 +317,7 @@ p9100attach(struct device *parent, struct device *self, void *args) * XXX there should be a rasops "clear margins" feature */ fbwscons_init(&sc->sc_sunfb, - isconsole && (sc->sc_sunfb.sf_width != 800)); + isconsole && (sc->sc_sunfb.sf_width >= 1024)); fbwscons_setcolormap(&sc->sc_sunfb, p9100_setcolor); /* @@ -497,7 +343,7 @@ p9100attach(struct device *parent, struct device *self, void *args) p9100_burner(sc, 1, 0); if (isconsole) { - if (sc->sc_sunfb.sf_width == 800) + if (sc->sc_sunfb.sf_width < 1024) row = 0; /* screen has been cleared above */ else row = -1; @@ -690,12 +536,12 @@ p9100_loadcmap_immediate(struct p9100_softc *sc, u_int start, u_int ncolors) u_char *p; P9100_SELECT_DAC(sc); - sc->sc_ctl->ctl_dac.pwraddr = start << 16; + P9100_WRITE_RAMDAC(sc, IBM525_WRADDR, start << 16); P9100_FLUSH_DAC(sc); for (p = sc->sc_cmap.cm_map[start], ncolors *= 3; ncolors-- > 0; p++) { P9100_SELECT_DAC(sc); - sc->sc_ctl->ctl_dac.paldata = (*p) << 16; + P9100_WRITE_RAMDAC(sc, IBM525_DATA, (*p) << 16); P9100_FLUSH_DAC(sc); } } @@ -703,11 +549,11 @@ p9100_loadcmap_immediate(struct p9100_softc *sc, u_int start, u_int ncolors) static __inline__ void p9100_loadcmap_deferred(struct p9100_softc *sc, u_int start, u_int ncolors) { - /* Schedule an interrupt for next retrace */ P9100_SELECT_SCR(sc); - sc->sc_ctl->ctl_scr.ier = IER_MASTER_ENABLE | IER_MASTER_INTERRUPT | - IER_VBLANK_ENABLE | IER_VBLANK_INTERRUPT; + P9100_WRITE_CTL(sc, P9000_INTERRUPT_ENABLE, + IER_MASTER_ENABLE | IER_MASTER_INTERRUPT | + IER_VBLANK_ENABLE | IER_VBLANK_INTERRUPT); } void @@ -719,13 +565,12 @@ p9100_burner(void *v, u_int on, u_int flags) s = splhigh(); P9100_SELECT_VCR(sc); - vcr = sc->sc_ctl->ctl_vcr.srtc1; + vcr = P9100_READ_CTL(sc, P9000_SRTC1); if (on) vcr |= SRTC1_VIDEN; else vcr &= ~SRTC1_VIDEN; - /* XXX - what about WSDISPLAY_BURN_VBLANK? */ - sc->sc_ctl->ctl_vcr.srtc1 = vcr; + P9100_WRITE_CTL(sc, P9000_SRTC1, vcr); #if NTCTRL > 0 tadpole_set_video(on); #endif @@ -737,12 +582,13 @@ p9100_intr(void *v) { struct p9100_softc *sc = v; - if (sc->sc_ctl->ctl_scr.ir & IER_VBLANK_INTERRUPT) { + if (P9100_READ_CTL(sc, P9000_INTERRUPT) & IER_VBLANK_INTERRUPT) { p9100_loadcmap_immediate(sc, 0, 256); /* Disable further interrupts now */ /* P9100_SELECT_SCR(sc); */ - sc->sc_ctl->ctl_scr.ier = IER_MASTER_ENABLE | 0; + P9100_WRITE_CTL(sc, P9000_INTERRUPT_ENABLE, + IER_MASTER_ENABLE | 0); return (1); } @@ -759,38 +605,39 @@ static __inline__ void p9100_drain(struct p9100_softc *); static __inline__ void p9100_drain(struct p9100_softc *sc) { - while (sc->sc_cmd->pe.status & (STATUS_QUAD_BUSY | STATUS_BLIT_BUSY)); + while (P9100_READ_CMD(sc, P9000_PE_STATUS) & + (STATUS_QUAD_BUSY | STATUS_BLIT_BUSY)); } void p9100_ras_init(struct p9100_softc *sc) { - /* * Setup safe defaults for the parameter and drawing engine, in * order to minimize the operations to do for ri_ops. */ P9100_SELECT_DE_LOW(sc); - sc->sc_cmd->de.drawmode = DM_PICK_CONTROL | 0 | - DM_BUFFER_CONTROL | DM_BUFFER_ENABLE0; + P9100_WRITE_CMD(sc, P9000_DE_DRAWMODE, + DM_PICK_CONTROL | 0 | DM_BUFFER_CONTROL | DM_BUFFER_ENABLE0); - sc->sc_cmd->de.patternx = 0; - sc->sc_cmd->de.patterny = 0; - sc->sc_cmd->de.planemask = 0xffffffff; /* enable all planes */ + P9100_WRITE_CMD(sc, P9000_DE_PATTERN_ORIGIN_X, 0); + P9100_WRITE_CMD(sc, P9000_DE_PATTERN_ORIGIN_Y, 0); + /* enable all planes */ + P9100_WRITE_CMD(sc, P9000_DE_PLANEMASK, 0xffffffff); /* Unclip */ - sc->sc_cmd->de.winmin = 0; - sc->sc_cmd->de.winmax = - P9100_COORDS(sc->sc_sunfb.sf_width - 1, sc->sc_sunfb.sf_height - 1); + P9100_WRITE_CMD(sc, P9000_DE_WINMIN, 0); + P9100_WRITE_CMD(sc, P9000_DE_WINMAX, + P9000_COORDS(sc->sc_sunfb.sf_width - 1, sc->sc_sunfb.sf_height - 1)); P9100_SELECT_DE_HIGH(sc); - sc->sc_cmd->de.bwinmin = 0; - sc->sc_cmd->de.bwinmax = - P9100_COORDS(sc->sc_sunfb.sf_width - 1, sc->sc_sunfb.sf_height - 1); + P9100_WRITE_CMD(sc, P9100_DE_B_WINMIN, 0); + P9100_WRITE_CMD(sc, P9100_DE_B_WINMAX, + P9000_COORDS(sc->sc_sunfb.sf_width - 1, sc->sc_sunfb.sf_height - 1)); P9100_SELECT_PE(sc); - sc->sc_cmd->pe.winoffset = 0; + P9100_WRITE_CMD(sc, P9000_PE_WINOFFSET, 0); } void @@ -810,18 +657,21 @@ p9100_ras_copycols(void *v, int row, int src, int dst, int n) p9100_drain(sc); P9100_SELECT_DE_LOW(sc); - sc->sc_cmd->de.raster = P9100_RASTER_SRC & P9100_RASTER_MASK; - - P9100_SELECT_COORD(sc,dc.coord[0]); - sc->sc_cmd->dc.coord[0].abs_x16y16 = P9100_COORDS(src, row); - sc->sc_cmd->dc.coord[1].abs_x16y16 = - P9100_COORDS(src + n, row + ri->ri_font->fontheight - 1); - P9100_SELECT_COORD(sc,dc.coord[2]); - sc->sc_cmd->dc.coord[2].abs_x16y16 = P9100_COORDS(dst, row); - sc->sc_cmd->dc.coord[3].abs_x16y16 = - P9100_COORDS(dst + n, row + ri->ri_font->fontheight - 1); - - sc->sc_junk = sc->sc_cmd->pe.blit; + P9100_WRITE_CMD(sc, P9000_DE_RASTER, + P9100_RASTER_SRC & P9100_RASTER_MASK); + + P9100_SELECT_COORD(sc, P9000_DC_COORD(0)); + P9100_WRITE_CMD(sc, P9000_DC_COORD(0) + P9000_COORD_XY, + P9000_COORDS(src, row)); + P9100_WRITE_CMD(sc, P9000_DC_COORD(1) + P9000_COORD_XY, + P9000_COORDS(src + n, row + ri->ri_font->fontheight - 1)); + P9100_SELECT_COORD(sc, P9000_DC_COORD(2)); + P9100_WRITE_CMD(sc, P9000_DC_COORD(2) + P9000_COORD_XY, + P9000_COORDS(dst, row)); + P9100_WRITE_CMD(sc, P9000_DC_COORD(3) + P9000_COORD_XY, + P9000_COORDS(dst + n, row + ri->ri_font->fontheight - 1)); + + sc->sc_junk = P9100_READ_CMD(sc, P9000_PE_BLIT); p9100_drain(sc); } @@ -841,18 +691,21 @@ p9100_ras_copyrows(void *v, int src, int dst, int n) p9100_drain(sc); P9100_SELECT_DE_LOW(sc); - sc->sc_cmd->de.raster = P9100_RASTER_SRC & P9100_RASTER_MASK; - - P9100_SELECT_COORD(sc,dc.coord[0]); - sc->sc_cmd->dc.coord[0].abs_x16y16 = P9100_COORDS(ri->ri_xorigin, src); - sc->sc_cmd->dc.coord[1].abs_x16y16 = - P9100_COORDS(ri->ri_xorigin + ri->ri_emuwidth - 1, src + n); - P9100_SELECT_COORD(sc,dc.coord[2]); - sc->sc_cmd->dc.coord[2].abs_x16y16 = P9100_COORDS(ri->ri_xorigin, dst); - sc->sc_cmd->dc.coord[3].abs_x16y16 = - P9100_COORDS(ri->ri_xorigin + ri->ri_emuwidth - 1, dst + n); - - sc->sc_junk = sc->sc_cmd->pe.blit; + P9100_WRITE_CMD(sc, P9000_DE_RASTER, + P9100_RASTER_SRC & P9100_RASTER_MASK); + + P9100_SELECT_COORD(sc, P9000_DC_COORD(0)); + P9100_WRITE_CMD(sc, P9000_DC_COORD(0) + P9000_COORD_XY, + P9000_COORDS(ri->ri_xorigin, src)); + P9100_WRITE_CMD(sc, P9000_DC_COORD(1) + P9000_COORD_XY, + P9000_COORDS(ri->ri_xorigin + ri->ri_emuwidth - 1, src + n)); + P9100_SELECT_COORD(sc, P9000_DC_COORD(2)); + P9100_WRITE_CMD(sc, P9000_DC_COORD(2) + P9000_COORD_XY, + P9000_COORDS(ri->ri_xorigin, dst)); + P9100_WRITE_CMD(sc, P9000_DC_COORD(3) + P9000_COORD_XY, + P9000_COORDS(ri->ri_xorigin + ri->ri_emuwidth - 1, dst + n)); + + sc->sc_junk = P9100_READ_CMD(sc, P9000_PE_BLIT); p9100_drain(sc); } @@ -874,15 +727,17 @@ p9100_ras_erasecols(void *v, int row, int col, int n, long int attr) p9100_drain(sc); P9100_SELECT_DE_LOW(sc); - sc->sc_cmd->de.raster = P9100_RASTER_PATTERN & P9100_RASTER_MASK; - sc->sc_cmd->de.color0 = P9100_COLOR8(bg); + P9100_WRITE_CMD(sc, P9000_DE_RASTER, + P9100_RASTER_PATTERN & P9100_RASTER_MASK); + P9100_WRITE_CMD(sc, P9100_DE_COLOR0, P9100_COLOR8(bg)); - P9100_SELECT_COORD(sc,lc.rect); - sc->sc_cmd->lc.rect.abs_x16y16 = P9100_COORDS(col, row); - sc->sc_cmd->lc.rect.abs_x16y16 = - P9100_COORDS(col + n, row + ri->ri_font->fontheight); + P9100_SELECT_COORD(sc, P9000_LC_RECT); + P9100_WRITE_CMD(sc, P9000_LC_RECT + P9000_COORD_XY, + P9000_COORDS(col, row)); + P9100_WRITE_CMD(sc, P9000_LC_RECT + P9000_COORD_XY, + P9000_COORDS(col + n, row + ri->ri_font->fontheight)); - sc->sc_junk = sc->sc_cmd->pe.quad; + sc->sc_junk = P9100_READ_CMD(sc, P9000_PE_QUAD); p9100_drain(sc); } @@ -898,26 +753,28 @@ p9100_ras_eraserows(void *v, int row, int n, long int attr) p9100_drain(sc); P9100_SELECT_DE_LOW(sc); - sc->sc_cmd->de.raster = P9100_RASTER_PATTERN & P9100_RASTER_MASK; - sc->sc_cmd->de.color0 = P9100_COLOR8(bg); + P9100_WRITE_CMD(sc, P9000_DE_RASTER, + P9100_RASTER_PATTERN & P9100_RASTER_MASK); + P9100_WRITE_CMD(sc, P9100_DE_COLOR0, P9100_COLOR8(bg)); - P9100_SELECT_COORD(sc,lc.rect); + P9100_SELECT_COORD(sc, P9000_LC_RECT); if (n == ri->ri_rows && ISSET(ri->ri_flg, RI_FULLCLEAR)) { - sc->sc_cmd->lc.rect.abs_x16y16 = P9100_COORDS(0, 0); - sc->sc_cmd->lc.rect.abs_x16y16 = - P9100_COORDS(ri->ri_width, ri->ri_height); + P9100_WRITE_CMD(sc, P9000_LC_RECT + P9000_COORD_XY, + P9000_COORDS(0, 0)); + P9100_WRITE_CMD(sc, P9000_LC_RECT + P9000_COORD_XY, + P9000_COORDS(ri->ri_width, ri->ri_height)); } else { n *= ri->ri_font->fontheight; row *= ri->ri_font->fontheight; row += ri->ri_yorigin; - sc->sc_cmd->lc.rect.abs_x16y16 = - P9100_COORDS(ri->ri_xorigin, row); - sc->sc_cmd->lc.rect.abs_x16y16 = - P9100_COORDS(ri->ri_xorigin + ri->ri_emuwidth, row + n); + P9100_WRITE_CMD(sc, P9000_LC_RECT + P9000_COORD_XY, + P9000_COORDS(ri->ri_xorigin, row)); + P9100_WRITE_CMD(sc, P9000_LC_RECT + P9000_COORD_XY, + P9000_COORDS(ri->ri_xorigin + ri->ri_emuwidth, row + n)); } - sc->sc_junk = sc->sc_cmd->pe.quad; + sc->sc_junk = P9100_READ_CMD(sc, P9000_PE_QUAD); p9100_drain(sc); } @@ -934,15 +791,17 @@ p9100_ras_do_cursor(struct rasops_info *ri) p9100_drain(sc); P9100_SELECT_DE_LOW(sc); - sc->sc_cmd->de.raster = (~P9100_RASTER_DST) & P9100_RASTER_MASK; + P9100_WRITE_CMD(sc, P9000_DE_RASTER, + (~P9100_RASTER_DST) & P9100_RASTER_MASK); - P9100_SELECT_COORD(sc,lc.rect); - sc->sc_cmd->lc.rect.abs_x16y16 = P9100_COORDS(col, row); - sc->sc_cmd->lc.rect.abs_x16y16 = - P9100_COORDS(col + ri->ri_font->fontwidth, - row + ri->ri_font->fontheight); + P9100_SELECT_COORD(sc, P9000_LC_RECT); + P9100_WRITE_CMD(sc, P9000_LC_RECT + P9000_COORD_XY, + P9000_COORDS(col, row)); + P9100_WRITE_CMD(sc, P9000_LC_RECT + P9000_COORD_XY, + P9000_COORDS(col + ri->ri_font->fontwidth, + row + ri->ri_font->fontheight)); - sc->sc_junk = sc->sc_cmd->pe.quad; + sc->sc_junk = P9100_READ_CMD(sc, P9000_PE_QUAD); p9100_drain(sc); } |