/* $OpenBSD: macfb.c,v 1.7 2006/01/09 21:52:12 miod Exp $ */ /* $NetBSD: macfb.c,v 1.11 2005/01/15 16:00:59 chs Exp $ */ /* * Copyright (c) 1998 Matt DeBergalis * 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. * 3. All advertising materials mentioning features or use of this software * must display the following acknowledgement: * This product includes software developed by Matt DeBergalis * 4. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission * * 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. */ #include #include #include #include #include #include #include #include #include #include #include #include #include struct cfdriver macfb_cd = { NULL, "macfb", DV_DULL }; int macfb_ioctl(void *, u_long, caddr_t, int, struct proc *); paddr_t macfb_mmap(void *, off_t, int); int macfb_alloc_screen(void *, const struct wsscreen_descr *, void **, int *, int *, long *); void macfb_free_screen(void *, void *); int macfb_show_screen(void *, void *, int, void (*)(void *, int, int), void *); const struct wsdisplay_accessops macfb_accessops = { macfb_ioctl, macfb_mmap, macfb_alloc_screen, macfb_free_screen, macfb_show_screen, NULL, /* load_font */ NULL, /* scrollback */ NULL, /* getchar */ NULL /* burner */ }; int macfb_alloc_attr(void *, int, int, int, long *); int macfb_init(struct macfb_devconfig *); int macfb_is_console(paddr_t); paddr_t macfb_consaddr; static struct macfb_devconfig macfb_console_dc; /* From Booter via locore */ extern long videoaddr; extern long videorowbytes; extern long videobitdepth; extern u_long videosize; extern u_int32_t mac68k_vidphys; extern u_int32_t mac68k_vidlen; int macfb_is_console(paddr_t addr) { if (addr != macfb_consaddr && (addr >= NBBASE && addr < NBTOP)) { /* * This is in the NuBus standard slot space range, so we * may well have to look at 0xFssxxxxx, too. Mask off the * slot number and duplicate it in bits 20-23, per IM:V * pp 459, 463, and IM:VI ch 30 p 17. * Note: this is an ugly hack and I wish I knew what * to do about it. -- sr */ addr = (paddr_t)(((u_long)addr & 0xff0fffff) | (((u_long)addr & 0x0f000000) >> 4)); } return ((mac68k_machine.serial_console & 0x03) == 0 && (addr == macfb_consaddr)); } int macfb_init(struct macfb_devconfig *dc) { struct rasops_info *ri = &dc->dc_ri; extern int rasops_alloc_cattr(void *, int, int, int, long *); /* * Clear display. We can't pass RI_CLEAR in ri_flg and have rasops * do it for us until we know how to setup the colormap first. */ memset((char *)dc->dc_vaddr + dc->dc_offset, dc->dc_depth <= 8 ? 0xff : 0, dc->dc_rowbytes * dc->dc_ht); bzero(ri, sizeof(*ri)); ri->ri_depth = dc->dc_depth; ri->ri_stride = dc->dc_rowbytes; ri->ri_flg = RI_CENTER; ri->ri_bits = (void *)(dc->dc_vaddr + dc->dc_offset); ri->ri_width = dc->dc_wid; ri->ri_height = dc->dc_ht; ri->ri_hw = dc; /* swap B and R */ if (ri->ri_depth == 16) { ri->ri_rnum = 5; ri->ri_rpos = 11; ri->ri_gnum = 6; ri->ri_gpos = 5; ri->ri_bnum = 5; ri->ri_bpos = 0; } /* * Ask for an unholy big display, rasops will trim this to more * reasonable values. */ if (rasops_init(ri, 160, 160) != 0) return (-1); if (ri->ri_depth <= 8) { /* * Until we know how to setup the colormap, constrain ourselves * to mono mode. Note that we need to use our own alloc_attr * routine to compensate for inverted black and white colors. */ ri->ri_ops.alloc_attr = macfb_alloc_attr; ri->ri_caps &= ~(WSSCREEN_WSCOLORS | WSSCREEN_HILIT); if (ri->ri_depth == 8) ri->ri_devcmap[15] = 0xffffffff; } strlcpy(dc->wsd.name, "std", sizeof(dc->wsd.name)); dc->wsd.ncols = ri->ri_cols; dc->wsd.nrows = ri->ri_rows; dc->wsd.textops = &ri->ri_ops; dc->wsd.fontwidth = ri->ri_font->fontwidth; dc->wsd.fontheight = ri->ri_font->fontheight; dc->wsd.capabilities = ri->ri_caps; return (0); } int macfb_alloc_attr(void *cookie, int fg, int bg, int flg, long *attr) { struct rasops_info *ri = cookie; int swap; if ((flg & (WSATTR_BLINK | WSATTR_HILIT | WSATTR_WSCOLORS)) != 0) return (EINVAL); /* * Default values are white on black. However, on indexed displays, * 0 is white and all bits set is black. */ if (ri->ri_depth <= 8) { fg = 0; bg = 15; } else { bg = WSCOL_BLACK; fg = WSCOL_WHITE; } if ((flg & WSATTR_REVERSE) != 0) { swap = fg; fg = bg; bg = swap; } *attr = (bg << 16) | (fg << 24) | ((flg & WSATTR_UNDERLINE) ? 7 : 6); return (0); } void macfb_attach_common(struct macfb_softc *sc, struct grfmode *gm) { struct wsemuldisplaydev_attach_args waa; struct wsscreen_descr *scrlist[1]; struct wsscreen_list screenlist; int isconsole; /* Print hardware characteristics. */ printf("%s: %dx%d ", sc->sc_dev.dv_xname, gm->width, gm->height); if (gm->psize == 1) printf("monochrome"); else printf("%d color", 1 << gm->psize); printf(" display\n"); isconsole = macfb_is_console(sc->sc_basepa + gm->fboff); if (isconsole) { sc->sc_dc = &macfb_console_dc; sc->sc_dc->nscreens = 1; } else { sc->sc_dc = malloc(sizeof(struct macfb_devconfig), M_DEVBUF, M_WAITOK); sc->sc_dc->dc_vaddr = (vaddr_t)gm->fbbase; sc->sc_dc->dc_paddr = sc->sc_basepa; sc->sc_dc->dc_size = gm->fbsize; sc->sc_dc->dc_wid = gm->width; sc->sc_dc->dc_ht = gm->height; sc->sc_dc->dc_depth = gm->psize; sc->sc_dc->dc_rowbytes = gm->rowbytes; sc->sc_dc->dc_offset = gm->fboff; sc->sc_dc->nscreens = 0; if (macfb_init(sc->sc_dc) != 0) return; } scrlist[0] = &sc->sc_dc->wsd; screenlist.nscreens = 1; screenlist.screens = (const struct wsscreen_descr **)scrlist; waa.console = isconsole; waa.scrdata = &screenlist; waa.accessops = &macfb_accessops; waa.accesscookie = sc; config_found((struct device *)sc, &waa, wsemuldisplaydevprint); } int macfb_ioctl(void *v, u_long cmd, caddr_t data, int flag, struct proc *p) { struct macfb_softc *sc = v; struct macfb_devconfig *dc = sc->sc_dc; struct wsdisplay_fbinfo *wdf; switch (cmd) { case WSDISPLAYIO_GTYPE: *(int *)data = WSDISPLAY_TYPE_MAC68K; break; case WSDISPLAYIO_GINFO: wdf = (struct wsdisplay_fbinfo *)data; wdf->height = dc->dc_ri.ri_height; wdf->width = dc->dc_ri.ri_width; wdf->depth = dc->dc_ri.ri_depth; wdf->cmsize = 0; /* until we can change it... */ break; case WSDISPLAYIO_LINEBYTES: *(u_int *)data = dc->dc_ri.ri_stride; break; case WSDISPLAYIO_GVIDEO: case WSDISPLAYIO_SVIDEO: break; default: return (-1); } return (0); } paddr_t macfb_mmap(void *v, off_t offset, int prot) { struct macfb_softc *sc = v; struct macfb_devconfig *dc = sc->sc_dc; paddr_t addr; if (offset >= 0 && offset < round_page(dc->dc_size)) addr = atop(dc->dc_paddr + dc->dc_offset + offset); else addr = (-1); return addr; } int macfb_alloc_screen(void *v, const struct wsscreen_descr *type, void **cookiep, int *curxp, int *curyp, long *defattrp) { struct macfb_softc *sc = v; struct rasops_info *ri = &sc->sc_dc->dc_ri; if (sc->sc_dc->nscreens > 0) return (ENOMEM); *cookiep = ri; *curxp = *curyp = 0; #ifdef notyet if ((ri->ri_caps & WSSCREEN_WSCOLORS) && ri->ri_depth <= 8) ri->ri_ops.alloc_attr(ri, WSCOL_WHITE, WSCOL_BLACK, WSATTR_WSCOLORS, defattrp); else #endif ri->ri_ops.alloc_attr(ri, 0, 0, 0, defattrp); sc->sc_dc->nscreens++; return (0); } void macfb_free_screen(void *v, void *cookie) { struct macfb_softc *sc = v; sc->sc_dc->nscreens--; } int macfb_show_screen(void *v, void *cookie, int waitok, void (*cb)(void *, int, int), void *cbarg) { return (0); } int macfb_cnattach(paddr_t addr) { struct macfb_devconfig *dc = &macfb_console_dc; long defattr; struct rasops_info *ri; dc->dc_vaddr = trunc_page(videoaddr); dc->dc_paddr = trunc_page(mac68k_vidphys); dc->dc_wid = videosize & 0xffff; dc->dc_ht = (videosize >> 16) & 0xffff; dc->dc_depth = videobitdepth; dc->dc_rowbytes = videorowbytes; dc->dc_size = (mac68k_vidlen > 0) ? mac68k_vidlen : dc->dc_ht * dc->dc_rowbytes; dc->dc_offset = m68k_page_offset(mac68k_vidphys); /* set up the display */ if (macfb_init(dc) != 0) return (-1); ri = &dc->dc_ri; #ifdef notyet if ((ri->ri_caps & WSSCREEN_WSCOLORS) && ri->ri_depth <= 8) ri->ri_ops.alloc_attr(ri, WSCOL_WHITE, WSCOL_BLACK, WSATTR_WSCOLORS, &defattr); else #endif ri->ri_ops.alloc_attr(ri, 0, 0, 0, &defattr); wsdisplay_cnattach(&dc->wsd, ri, 0, 0, defattr); macfb_consaddr = addr; return (0); }