1*86973f0aSmacallan /* $NetBSD: summitfb.c,v 1.30 2025/01/27 12:10:19 macallan Exp $ */ 29ad1548eSmacallan 39ad1548eSmacallan /* $OpenBSD: sti_pci.c,v 1.7 2009/02/06 22:51:04 miod Exp $ */ 49ad1548eSmacallan 59ad1548eSmacallan /* 69ad1548eSmacallan * Copyright (c) 2006, 2007 Miodrag Vallat. 79ad1548eSmacallan ^ 2024 Michael Lorenz 89ad1548eSmacallan * 99ad1548eSmacallan * Permission to use, copy, modify, and distribute this software for any 109ad1548eSmacallan * purpose with or without fee is hereby granted, provided that the above 119ad1548eSmacallan * copyright notice, this permission notice, and the disclaimer below 129ad1548eSmacallan * appear in all copies. 139ad1548eSmacallan * 149ad1548eSmacallan * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES 159ad1548eSmacallan * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 169ad1548eSmacallan * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR 179ad1548eSmacallan * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 189ad1548eSmacallan * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 199ad1548eSmacallan * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF 209ad1548eSmacallan * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 219ad1548eSmacallan */ 229ad1548eSmacallan 239ad1548eSmacallan /* 2412e4281eSmacallan * a native driver for HP Visualize FX graphics cards, so far tested only on 2512e4281eSmacallan * my FX4 269ad1548eSmacallan * STI portions are from Miodrag Vallat's sti_pci.c 279ad1548eSmacallan */ 289ad1548eSmacallan 29c84cdbddSriastradh #include <sys/cdefs.h> 30*86973f0aSmacallan __KERNEL_RCSID(0, "$NetBSD: summitfb.c,v 1.30 2025/01/27 12:10:19 macallan Exp $"); 31c84cdbddSriastradh 329ad1548eSmacallan #include <sys/param.h> 339ad1548eSmacallan #include <sys/systm.h> 349ad1548eSmacallan #include <sys/kmem.h> 359ad1548eSmacallan #include <sys/device.h> 369ad1548eSmacallan #include <sys/mutex.h> 379ad1548eSmacallan 389ad1548eSmacallan #include <dev/pci/pcivar.h> 399ad1548eSmacallan #include <dev/pci/pcireg.h> 409ad1548eSmacallan #include <dev/pci/pcidevs.h> 419ad1548eSmacallan #include <dev/pci/pciio.h> 429ad1548eSmacallan 439ad1548eSmacallan #include <dev/wscons/wsdisplayvar.h> 449ad1548eSmacallan #include <dev/wscons/wsconsio.h> 459ad1548eSmacallan #include <dev/wsfont/wsfont.h> 469ad1548eSmacallan #include <dev/rasops/rasops.h> 479ad1548eSmacallan #include <dev/wscons/wsdisplay_vconsvar.h> 489ad1548eSmacallan #include <dev/pci/wsdisplay_pci.h> 499ad1548eSmacallan #include <dev/wscons/wsdisplay_glyphcachevar.h> 509ad1548eSmacallan 519ad1548eSmacallan #include <dev/ic/stireg.h> 52e4bd382dSmacallan #include <dev/ic/summitreg.h> 539ad1548eSmacallan #include <dev/ic/stivar.h> 549ad1548eSmacallan 559ad1548eSmacallan #include "opt_summitfb.h" 569ad1548eSmacallan 579ad1548eSmacallan #ifdef SUMMITFB_DEBUG 58ebd58a30Sriastradh #define DPRINTF(s) printf s 599ad1548eSmacallan #else 60c84cdbddSriastradh #define DPRINTF(s) __nothing 619ad1548eSmacallan #endif 629ad1548eSmacallan 639ad1548eSmacallan int summitfb_match(device_t, cfdata_t, void *); 649ad1548eSmacallan void summitfb_attach(device_t, device_t, void *); 659ad1548eSmacallan 669ad1548eSmacallan struct summitfb_softc { 679ad1548eSmacallan device_t sc_dev; 689ad1548eSmacallan pci_chipset_tag_t sc_pc; 699ad1548eSmacallan pcitag_t sc_tag; 709ad1548eSmacallan 719ad1548eSmacallan /* stuff we need in order to use the STI ROM */ 729ad1548eSmacallan struct sti_softc sc_base; 739ad1548eSmacallan struct sti_screen sc_scr; 749ad1548eSmacallan bus_space_handle_t sc_romh; 759ad1548eSmacallan 769ad1548eSmacallan int sc_width, sc_height; 779ad1548eSmacallan int sc_locked; 789ad1548eSmacallan struct vcons_screen sc_console_screen; 799ad1548eSmacallan struct wsscreen_descr sc_defaultscreen_descr; 809ad1548eSmacallan const struct wsscreen_descr *sc_screens[1]; 819ad1548eSmacallan struct wsscreen_list sc_screenlist; 829ad1548eSmacallan struct vcons_data vd; 839ad1548eSmacallan int sc_mode; 849ad1548eSmacallan u_char sc_cmap_red[256]; 859ad1548eSmacallan u_char sc_cmap_green[256]; 869ad1548eSmacallan u_char sc_cmap_blue[256]; 875ac86b16Smacallan uint32_t sc_write_mode, sc_read_mode; 889ad1548eSmacallan /* cursor stuff */ 899ad1548eSmacallan int sc_cursor_x, sc_cursor_y; 909ad1548eSmacallan int sc_hot_x, sc_hot_y, sc_enabled; 917bbb04c1Smacallan /* font-in-vram */ 927bbb04c1Smacallan struct wsdisplay_font *sc_font; 937bbb04c1Smacallan int sc_font_start; /* x of font area */ 947bbb04c1Smacallan int sc_cols; /* chars per line in font area */ 95*86973f0aSmacallan uint32_t sc_palette[16]; 969ad1548eSmacallan int sc_video_on; 979ad1548eSmacallan glyphcache sc_gc; 989ad1548eSmacallan }; 999ad1548eSmacallan 1009ad1548eSmacallan CFATTACH_DECL_NEW(summitfb, sizeof(struct summitfb_softc), 1019ad1548eSmacallan summitfb_match, summitfb_attach, NULL, NULL); 1029ad1548eSmacallan 103c84cdbddSriastradh int summitfb_readbar(struct sti_softc *, struct pci_attach_args *, u_int, 104c84cdbddSriastradh int); 1059ad1548eSmacallan int summitfb_check_rom(struct summitfb_softc *, struct pci_attach_args *); 1069ad1548eSmacallan void summitfb_enable_rom(struct sti_softc *); 1079ad1548eSmacallan void summitfb_disable_rom(struct sti_softc *); 1089ad1548eSmacallan void summitfb_enable_rom_internal(struct summitfb_softc *); 1099ad1548eSmacallan void summitfb_disable_rom_internal(struct summitfb_softc *); 1109ad1548eSmacallan 1119ad1548eSmacallan void summitfb_setup(struct summitfb_softc *); 1129ad1548eSmacallan 1139ad1548eSmacallan /* XXX these really need to go into their own header */ 1149ad1548eSmacallan int sti_pci_is_console(struct pci_attach_args *, bus_addr_t *); 1159ad1548eSmacallan int sti_rom_setup(struct sti_rom *, bus_space_tag_t, bus_space_tag_t, 1169ad1548eSmacallan bus_space_handle_t, bus_addr_t *, u_int); 1179ad1548eSmacallan int sti_screen_setup(struct sti_screen *, int); 1189ad1548eSmacallan void sti_describe_screen(struct sti_softc *, struct sti_screen *); 1199ad1548eSmacallan 1209ad1548eSmacallan #define PCI_ROM_SIZE(mr) \ 1219ad1548eSmacallan (PCI_MAPREG_ROM_ADDR(mr) & -PCI_MAPREG_ROM_ADDR(mr)) 1229ad1548eSmacallan 1239ad1548eSmacallan /* wsdisplay stuff */ 1249ad1548eSmacallan static int summitfb_ioctl(void *, void *, u_long, void *, int, 1259ad1548eSmacallan struct lwp *); 1269ad1548eSmacallan static paddr_t summitfb_mmap(void *, void *, off_t, int); 127c84cdbddSriastradh static void summitfb_init_screen(void *, struct vcons_screen *, int, 128c84cdbddSriastradh long *); 1299ad1548eSmacallan 130c84cdbddSriastradh static int summitfb_putcmap(struct summitfb_softc *, 131c84cdbddSriastradh struct wsdisplay_cmap *); 132c84cdbddSriastradh static int summitfb_getcmap(struct summitfb_softc *, 133c84cdbddSriastradh struct wsdisplay_cmap *); 1349ad1548eSmacallan static void summitfb_restore_palette(struct summitfb_softc *); 1359ad1548eSmacallan static int summitfb_putpalreg(struct summitfb_softc *, uint8_t, uint8_t, 1369ad1548eSmacallan uint8_t, uint8_t); 1379ad1548eSmacallan 1389ad1548eSmacallan static inline void summitfb_setup_fb(struct summitfb_softc *); 139d2ce6272Smacallan static void summitfb_clearfb(struct summitfb_softc *); 1409ad1548eSmacallan static void summitfb_rectfill(struct summitfb_softc *, int, int, int, int, 1419ad1548eSmacallan uint32_t); 1429ad1548eSmacallan static void summitfb_bitblt(void *, int, int, int, int, int, 1439ad1548eSmacallan int, int); 1449ad1548eSmacallan 1459ad1548eSmacallan static void summitfb_cursor(void *, int, int, int); 1469ad1548eSmacallan static void summitfb_putchar(void *, int, int, u_int, long); 1477bbb04c1Smacallan static void summitfb_putchar_fast(void *, int, int, u_int, long); 1487bbb04c1Smacallan static void summitfb_loadfont(struct summitfb_softc *); 1499ad1548eSmacallan static void summitfb_putchar_aa(void *, int, int, u_int, long); 1509ad1548eSmacallan static void summitfb_copycols(void *, int, int, int, int); 1519ad1548eSmacallan static void summitfb_erasecols(void *, int, int, int, long); 1529ad1548eSmacallan static void summitfb_copyrows(void *, int, int, int); 1539ad1548eSmacallan static void summitfb_eraserows(void *, int, int, long); 1549ad1548eSmacallan 1559ad1548eSmacallan static void summitfb_move_cursor(struct summitfb_softc *, int, int); 156c84cdbddSriastradh static int summitfb_do_cursor(struct summitfb_softc *, 157c84cdbddSriastradh struct wsdisplay_cursor *); 1589ad1548eSmacallan 1599ad1548eSmacallan static void summitfb_set_video(struct summitfb_softc *, int); 1609ad1548eSmacallan 1616fcfb3adSmacallan static void summitfb_copyfont(struct summitfb_softc *); 1626fcfb3adSmacallan 1639ad1548eSmacallan struct wsdisplay_accessops summitfb_accessops = { 164c84cdbddSriastradh .ioctl = summitfb_ioctl, 165c84cdbddSriastradh .mmap = summitfb_mmap, 166c84cdbddSriastradh .alloc_screen = NULL, 167c84cdbddSriastradh .free_screen = NULL, 168c84cdbddSriastradh .show_screen = NULL, 169c84cdbddSriastradh .load_font = NULL, 170c84cdbddSriastradh .pollc = NULL, 171c84cdbddSriastradh .scroll = NULL, 1729ad1548eSmacallan }; 1739ad1548eSmacallan 1749ad1548eSmacallan static inline void summitfb_wait_fifo(struct summitfb_softc *, uint32_t); 175e74ab3bcSmacallan static inline void summitfb_wait(struct summitfb_softc *); 1769ad1548eSmacallan 1776fcfb3adSmacallan int sti_fetchfonts(struct sti_screen *, struct sti_inqconfout *, uint32_t, 1786fcfb3adSmacallan u_int); 1796fcfb3adSmacallan 1809ad1548eSmacallan int 1819ad1548eSmacallan summitfb_match(device_t parent, cfdata_t cf, void *aux) 1829ad1548eSmacallan { 1839ad1548eSmacallan struct pci_attach_args *paa = aux; 1849ad1548eSmacallan 1859ad1548eSmacallan if (PCI_VENDOR(paa->pa_id) != PCI_VENDOR_HP) 1869ad1548eSmacallan return 0; 1879ad1548eSmacallan 1889ad1548eSmacallan if (PCI_PRODUCT(paa->pa_id) == PCI_PRODUCT_HP_VISUALIZE_FX4) 1899ad1548eSmacallan return 10; /* beat out sti at pci */ 1909ad1548eSmacallan 1919ad1548eSmacallan return 0; 1929ad1548eSmacallan } 1939ad1548eSmacallan 1949ad1548eSmacallan static inline uint32_t 1959ad1548eSmacallan summitfb_read4(struct summitfb_softc *sc, uint32_t offset) 1969ad1548eSmacallan { 1979ad1548eSmacallan struct sti_rom *rom = sc->sc_base.sc_rom; 1989ad1548eSmacallan bus_space_tag_t memt = rom->memt; 1999ad1548eSmacallan bus_space_handle_t memh = rom->regh[2]; 200c84cdbddSriastradh 2019ad1548eSmacallan return bus_space_read_stream_4(memt, memh, offset - 0x400000); 2029ad1548eSmacallan } 2039ad1548eSmacallan 2049ad1548eSmacallan static inline void 2059ad1548eSmacallan summitfb_write4(struct summitfb_softc *sc, uint32_t offset, uint32_t val) 2069ad1548eSmacallan { 2079ad1548eSmacallan struct sti_rom *rom = sc->sc_base.sc_rom; 2089ad1548eSmacallan bus_space_tag_t memt = rom->memt; 2099ad1548eSmacallan bus_space_handle_t memh = rom->regh[2]; 210c84cdbddSriastradh 2119ad1548eSmacallan bus_space_write_stream_4(memt, memh, offset - 0x400000, val); 2129ad1548eSmacallan } 2139ad1548eSmacallan 214*86973f0aSmacallan static inline void 215*86973f0aSmacallan summitfb_write_mode(struct summitfb_softc *sc, uint32_t mode) 216*86973f0aSmacallan { 217*86973f0aSmacallan if (sc->sc_write_mode == mode) 218*86973f0aSmacallan return; 219*86973f0aSmacallan summitfb_wait(sc); 220*86973f0aSmacallan summitfb_write4(sc, VISFX_VRAM_WRITE_MODE, mode); 221*86973f0aSmacallan sc->sc_write_mode = mode; 222*86973f0aSmacallan } 223*86973f0aSmacallan 224*86973f0aSmacallan static inline void 225*86973f0aSmacallan summitfb_read_mode(struct summitfb_softc *sc, uint32_t mode) 226*86973f0aSmacallan { 227*86973f0aSmacallan if (sc->sc_read_mode == mode) 228*86973f0aSmacallan return; 229*86973f0aSmacallan summitfb_wait(sc); 230*86973f0aSmacallan summitfb_write4(sc, VISFX_VRAM_READ_MODE, mode); 231*86973f0aSmacallan sc->sc_read_mode = mode; 232*86973f0aSmacallan } 233*86973f0aSmacallan 2349ad1548eSmacallan void 2359ad1548eSmacallan summitfb_attach(device_t parent, device_t self, void *aux) 2369ad1548eSmacallan { 2379ad1548eSmacallan struct summitfb_softc *sc = device_private(self); 2389ad1548eSmacallan struct pci_attach_args *paa = aux; 2399ad1548eSmacallan struct sti_rom *rom; 2409ad1548eSmacallan struct rasops_info *ri; 2419ad1548eSmacallan struct wsemuldisplaydev_attach_args aa; 242d7adffb3Smacallan struct sti_dd *dd; 2439ad1548eSmacallan unsigned long defattr = 0; 2449ad1548eSmacallan int ret, is_console = 0; 2459ad1548eSmacallan 2469ad1548eSmacallan sc->sc_dev = self; 2479ad1548eSmacallan 2489ad1548eSmacallan sc->sc_pc = paa->pa_pc; 2499ad1548eSmacallan sc->sc_tag = paa->pa_tag; 2509ad1548eSmacallan sc->sc_base.sc_dev = self; 2519ad1548eSmacallan sc->sc_base.sc_enable_rom = summitfb_enable_rom; 2529ad1548eSmacallan sc->sc_base.sc_disable_rom = summitfb_disable_rom; 2539ad1548eSmacallan 2549ad1548eSmacallan aprint_normal("\n"); 2559ad1548eSmacallan 2569ad1548eSmacallan if (summitfb_check_rom(sc, paa) != 0) 2579ad1548eSmacallan return; 2589ad1548eSmacallan 2599ad1548eSmacallan ret = sti_pci_is_console(paa, sc->sc_base. bases); 2609ad1548eSmacallan if (ret != 0) { 2619ad1548eSmacallan sc->sc_base.sc_flags |= STI_CONSOLE; 2629ad1548eSmacallan is_console = 1; 2639ad1548eSmacallan } 264c84cdbddSriastradh rom = kmem_zalloc(sizeof(*rom), KM_SLEEP); 2659ad1548eSmacallan rom->rom_softc = &sc->sc_base; 2669ad1548eSmacallan ret = sti_rom_setup(rom, paa->pa_iot, paa->pa_memt, sc->sc_romh, 2679ad1548eSmacallan sc->sc_base.bases, STI_CODEBASE_MAIN); 2689ad1548eSmacallan if (ret != 0) { 2699ad1548eSmacallan kmem_free(rom, sizeof(*rom)); 2709ad1548eSmacallan return; 2719ad1548eSmacallan } 2729ad1548eSmacallan 2739ad1548eSmacallan sc->sc_base.sc_rom = rom; 274d7adffb3Smacallan dd = &rom->rom_dd; 2759ad1548eSmacallan 2769ad1548eSmacallan sc->sc_scr.scr_rom = sc->sc_base.sc_rom; 2779ad1548eSmacallan ret = sti_screen_setup(&sc->sc_scr, STI_FBMODE); 27838fdde35Sskrll 2796fcfb3adSmacallan sti_fetchfonts(&sc->sc_scr, NULL, dd->dd_fntaddr, 0); 280d7adffb3Smacallan wsfont_init(); 2816fcfb3adSmacallan summitfb_copyfont(sc); 28238fdde35Sskrll 2839ad1548eSmacallan sc->sc_width = sc->sc_scr.scr_cfg.scr_width; 2849ad1548eSmacallan sc->sc_height = sc->sc_scr.scr_cfg.scr_height; 285e74ab3bcSmacallan sc->sc_write_mode = 0xffffffff; 2865ac86b16Smacallan sc->sc_read_mode = 0xffffffff; 2879ad1548eSmacallan 2889ad1548eSmacallan #ifdef SUMMITFB_DEBUG 2899ad1548eSmacallan sc->sc_height -= 200; 2909ad1548eSmacallan #endif 2919ad1548eSmacallan 2929ad1548eSmacallan sc->sc_defaultscreen_descr = (struct wsscreen_descr){ 293c84cdbddSriastradh .name = "default", 294c84cdbddSriastradh .ncols = 0, .nrows = 0, 295c84cdbddSriastradh .textops = NULL, 296c84cdbddSriastradh .fontwidth = 8, .fontheight = 16, 297c84cdbddSriastradh .capabilities = WSSCREEN_WSCOLORS | WSSCREEN_HILIT | 298c84cdbddSriastradh WSSCREEN_UNDERLINE | WSSCREEN_RESIZE, 299c84cdbddSriastradh .modecookie = NULL, 3009ad1548eSmacallan }; 3019ad1548eSmacallan 3029ad1548eSmacallan sc->sc_screens[0] = &sc->sc_defaultscreen_descr; 3039ad1548eSmacallan sc->sc_screenlist = (struct wsscreen_list){1, sc->sc_screens}; 3049ad1548eSmacallan sc->sc_mode = WSDISPLAYIO_MODE_EMUL; 3059ad1548eSmacallan sc->sc_locked = 0; 3069ad1548eSmacallan 3079ad1548eSmacallan vcons_init(&sc->vd, sc, &sc->sc_defaultscreen_descr, 3089ad1548eSmacallan &summitfb_accessops); 3099ad1548eSmacallan sc->vd.init_screen = summitfb_init_screen; 3109ad1548eSmacallan sc->vd.show_screen_cookie = &sc->sc_gc; 3119ad1548eSmacallan sc->vd.show_screen_cb = glyphcache_adapt; 3129ad1548eSmacallan ri = &sc->sc_console_screen.scr_ri; 3139ad1548eSmacallan 3149ad1548eSmacallan sc->sc_gc.gc_bitblt = summitfb_bitblt; 3159ad1548eSmacallan sc->sc_gc.gc_blitcookie = sc; 3169ad1548eSmacallan sc->sc_gc.gc_rop = RopSrc; 3177bbb04c1Smacallan 3187bbb04c1Smacallan summitfb_setup(sc); 3199ad1548eSmacallan 3209ad1548eSmacallan vcons_init_screen(&sc->vd, &sc->sc_console_screen, 1, &defattr); 3219ad1548eSmacallan sc->sc_console_screen.scr_flags |= VCONS_SCREEN_IS_STATIC; 3229ad1548eSmacallan 3239ad1548eSmacallan sc->sc_defaultscreen_descr.textops = &ri->ri_ops; 3249ad1548eSmacallan sc->sc_defaultscreen_descr.capabilities = ri->ri_caps; 3259ad1548eSmacallan sc->sc_defaultscreen_descr.nrows = ri->ri_rows; 3269ad1548eSmacallan sc->sc_defaultscreen_descr.ncols = ri->ri_cols; 3279ad1548eSmacallan 3287bbb04c1Smacallan /* 3297bbb04c1Smacallan * STI lies to us - it reports a 2048x2048 framebuffer but blitter 3307bbb04c1Smacallan * ops wrap around below 1024 and we seem to have only about 250 3317bbb04c1Smacallan * usable columns to the right. Should still be enough to cache 3327bbb04c1Smacallan * a font or four. 3337bbb04c1Smacallan * So, the framebuffer seems to be 1536x1024, which is odd since the 3347bbb04c1Smacallan * FX4 is supposed to support resolutions higher than 1280x1024. 3357bbb04c1Smacallan * I guess video memory is allocated in 512x512 chunks 3367bbb04c1Smacallan */ 3375ac86b16Smacallan glyphcache_init(&sc->sc_gc, 3387bbb04c1Smacallan sc->sc_height, 3395ac86b16Smacallan sc->sc_height, 3405ac86b16Smacallan (sc->sc_width + 511) & (~511), 3419ad1548eSmacallan ri->ri_font->fontwidth, 3429ad1548eSmacallan ri->ri_font->fontheight, 3439ad1548eSmacallan defattr); 3449ad1548eSmacallan 3459ad1548eSmacallan summitfb_restore_palette(sc); 3469ad1548eSmacallan summitfb_rectfill(sc, 0, 0, sc->sc_width, sc->sc_height, 3479ad1548eSmacallan ri->ri_devcmap[(defattr >> 16) & 0xff]); 3489ad1548eSmacallan summitfb_setup_fb(sc); 3499ad1548eSmacallan 3509ad1548eSmacallan if (is_console) { 3519ad1548eSmacallan wsdisplay_cnattach(&sc->sc_defaultscreen_descr, ri, 0, 0, 3529ad1548eSmacallan defattr); 3539ad1548eSmacallan 3549ad1548eSmacallan vcons_replay_msgbuf(&sc->sc_console_screen); 3559ad1548eSmacallan } 3569ad1548eSmacallan 357b04ca5b4Smacallan aprint_normal_dev(sc->sc_dev, "%s at %dx%d\n", sc->sc_scr.name, 358b04ca5b4Smacallan sc->sc_width, sc->sc_height); 359b04ca5b4Smacallan 3609ad1548eSmacallan /* no suspend/resume support yet */ 361c84cdbddSriastradh pmf_device_register(sc->sc_dev, NULL, NULL); 3629ad1548eSmacallan 3639ad1548eSmacallan aa.console = is_console; 3649ad1548eSmacallan aa.scrdata = &sc->sc_screenlist; 3659ad1548eSmacallan aa.accessops = &summitfb_accessops; 3669ad1548eSmacallan aa.accesscookie = &sc->vd; 3679ad1548eSmacallan 3689ad1548eSmacallan config_found(sc->sc_dev, &aa, wsemuldisplaydevprint, CFARGS_NONE); 369*86973f0aSmacallan #ifdef DEBUG 370*86973f0aSmacallan { 371*86973f0aSmacallan int i; 372*86973f0aSmacallan 373*86973f0aSmacallan summitfb_rectfill(sc, 0, 824, 1280, 200, 0x00); 374*86973f0aSmacallan summitfb_rectfill(sc, 5, 830, 100, 190, 0xe0); /* red */ 375*86973f0aSmacallan 376*86973f0aSmacallan summitfb_write_mode(sc, OTC01 | BIN332F | BUFovl); 377*86973f0aSmacallan summitfb_read_mode(sc, OTC01 | BIN332F | BUFovl); 378*86973f0aSmacallan summitfb_write4(sc, VISFX_CBR, 0x3f0000ff); 379*86973f0aSmacallan summitfb_write4(sc, VISFX_FG_COLOUR, 0x7f00ff00); 380*86973f0aSmacallan 381*86973f0aSmacallan for (i = 0; i < 16; i++) { 382*86973f0aSmacallan summitfb_wait_fifo(sc, 10); 383*86973f0aSmacallan summitfb_write4(sc, VISFX_IBO, 0x200 | (14 << 4) | i); 384*86973f0aSmacallan summitfb_write4(sc, VISFX_COPY_SRC, (5 << 16) | 830); 385*86973f0aSmacallan summitfb_write4(sc, VISFX_COPY_WH, (30 << 16) | 90); 386*86973f0aSmacallan summitfb_write4(sc, VISFX_COPY_DST, 387*86973f0aSmacallan ((i * 35 + 200) << 16) | 830); 388*86973f0aSmacallan } 389*86973f0aSmacallan summitfb_write_mode(sc, OTC01 | BIN332F | BUFovl | 0x8c0); 390*86973f0aSmacallan summitfb_write4(sc, VISFX_FG_COLOUR, 0x7f00ff00); 391*86973f0aSmacallan for (i = 0; i < 16; i++) { 392*86973f0aSmacallan summitfb_wait_fifo(sc, 10); 393*86973f0aSmacallan summitfb_write4(sc, VISFX_IBO, 0x200 | (i << 4) | 5); 394*86973f0aSmacallan summitfb_write4(sc, VISFX_START, 395*86973f0aSmacallan ((i * 35 + 200) << 16) | 930); 396*86973f0aSmacallan summitfb_write4(sc, VISFX_SIZE, (30 << 16) | 90); 397*86973f0aSmacallan } 398*86973f0aSmacallan } 399*86973f0aSmacallan #endif 4009ad1548eSmacallan } 4019ad1548eSmacallan 4029ad1548eSmacallan /* 4039ad1548eSmacallan * Grovel the STI ROM image. 4049ad1548eSmacallan */ 4059ad1548eSmacallan int 4069ad1548eSmacallan summitfb_check_rom(struct summitfb_softc *spc, struct pci_attach_args *pa) 4079ad1548eSmacallan { 4089ad1548eSmacallan struct sti_softc *sc = &spc->sc_base; 4099ad1548eSmacallan pcireg_t address, mask; 4109ad1548eSmacallan bus_space_handle_t romh; 4119ad1548eSmacallan bus_size_t romsize, subsize, stiromsize; 4129ad1548eSmacallan bus_addr_t selected, offs, suboffs; 4139ad1548eSmacallan uint32_t tmp; 4149ad1548eSmacallan int i; 4159ad1548eSmacallan int rc; 4169ad1548eSmacallan 4179ad1548eSmacallan /* sort of inline sti_pci_enable_rom(sc) */ 4189ad1548eSmacallan address = pci_conf_read(pa->pa_pc, pa->pa_tag, PCI_MAPREG_ROM); 4199ad1548eSmacallan pci_conf_write(pa->pa_pc, pa->pa_tag, PCI_MAPREG_ROM, 4209ad1548eSmacallan ~PCI_MAPREG_ROM_ENABLE); 4219ad1548eSmacallan mask = pci_conf_read(pa->pa_pc, pa->pa_tag, PCI_MAPREG_ROM); 4229ad1548eSmacallan address |= PCI_MAPREG_ROM_ENABLE; 4239ad1548eSmacallan pci_conf_write(pa->pa_pc, pa->pa_tag, PCI_MAPREG_ROM, address); 4249ad1548eSmacallan sc->sc_flags |= STI_ROM_ENABLED; 425c84cdbddSriastradh 4269ad1548eSmacallan /* 4279ad1548eSmacallan * Map the complete ROM for now. 4289ad1548eSmacallan */ 4299ad1548eSmacallan romsize = PCI_ROM_SIZE(mask); 4309ad1548eSmacallan DPRINTF(("%s: mapping rom @ %lx for %lx\n", __func__, 4319ad1548eSmacallan (long)PCI_MAPREG_ROM_ADDR(address), (long)romsize)); 4329ad1548eSmacallan 4339ad1548eSmacallan rc = bus_space_map(pa->pa_memt, PCI_MAPREG_ROM_ADDR(address), romsize, 4349ad1548eSmacallan 0, &romh); 4359ad1548eSmacallan if (rc != 0) { 4369ad1548eSmacallan aprint_error_dev(sc->sc_dev, "can't map PCI ROM (%d)\n", rc); 4379ad1548eSmacallan goto fail2; 4389ad1548eSmacallan } 4399ad1548eSmacallan 4409ad1548eSmacallan summitfb_disable_rom_internal(spc); 44138fdde35Sskrll 4429ad1548eSmacallan /* 4439ad1548eSmacallan * Iterate over the ROM images, pick the best candidate. 4449ad1548eSmacallan */ 4459ad1548eSmacallan selected = (bus_addr_t)-1; 4469ad1548eSmacallan for (offs = 0; offs < romsize; offs += subsize) { 4479ad1548eSmacallan summitfb_enable_rom_internal(spc); 4489ad1548eSmacallan /* 4499ad1548eSmacallan * Check for a valid ROM header. 4509ad1548eSmacallan */ 4519ad1548eSmacallan tmp = bus_space_read_4(pa->pa_memt, romh, offs + 0); 4529ad1548eSmacallan tmp = le32toh(tmp); 4539ad1548eSmacallan if (tmp != 0x55aa0000) { 4549ad1548eSmacallan summitfb_disable_rom_internal(spc); 4559ad1548eSmacallan if (offs == 0) { 4569ad1548eSmacallan aprint_error_dev(sc->sc_dev, 457c84cdbddSriastradh "invalid PCI ROM header signature" 458c84cdbddSriastradh " (%08x)\n", tmp); 4599ad1548eSmacallan rc = EINVAL; 4609ad1548eSmacallan } 4619ad1548eSmacallan break; 4629ad1548eSmacallan } 4639ad1548eSmacallan 4649ad1548eSmacallan /* 4659ad1548eSmacallan * Check ROM type. 4669ad1548eSmacallan */ 4679ad1548eSmacallan tmp = bus_space_read_4(pa->pa_memt, romh, offs + 4); 4689ad1548eSmacallan tmp = le32toh(tmp); 4699ad1548eSmacallan if (tmp != 0x00000001) { /* 1 == STI ROM */ 4709ad1548eSmacallan summitfb_disable_rom_internal(spc); 4719ad1548eSmacallan if (offs == 0) { 4729ad1548eSmacallan aprint_error_dev(sc->sc_dev, 4739ad1548eSmacallan "invalid PCI ROM type (%08x)\n", tmp); 4749ad1548eSmacallan rc = EINVAL; 4759ad1548eSmacallan } 4769ad1548eSmacallan break; 4779ad1548eSmacallan } 4789ad1548eSmacallan 4799ad1548eSmacallan subsize = (bus_addr_t)bus_space_read_2(pa->pa_memt, romh, 4809ad1548eSmacallan offs + 0x0c); 4819ad1548eSmacallan subsize <<= 9; 4829ad1548eSmacallan 4839ad1548eSmacallan #ifdef SUMMITFB_DEBUG 4849ad1548eSmacallan summitfb_disable_rom_internal(spc); 4859ad1548eSmacallan DPRINTF(("ROM offset %08x size %08x type %08x", 4869ad1548eSmacallan (u_int)offs, (u_int)subsize, tmp)); 4879ad1548eSmacallan summitfb_enable_rom_internal(spc); 4889ad1548eSmacallan #endif 4899ad1548eSmacallan 4909ad1548eSmacallan /* 4919ad1548eSmacallan * Check for a valid ROM data structure. 4929ad1548eSmacallan * We do not need it except to know what architecture the ROM 4939ad1548eSmacallan * code is for. 4949ad1548eSmacallan */ 4959ad1548eSmacallan 49638fdde35Sskrll suboffs = offs + bus_space_read_2(pa->pa_memt, romh, 4979ad1548eSmacallan offs + 0x18); 4989ad1548eSmacallan tmp = bus_space_read_4(pa->pa_memt, romh, suboffs + 0); 4999ad1548eSmacallan tmp = le32toh(tmp); 5009ad1548eSmacallan if (tmp != 0x50434952) { /* PCIR */ 5019ad1548eSmacallan summitfb_disable_rom_internal(spc); 5029ad1548eSmacallan if (offs == 0) { 5039ad1548eSmacallan aprint_error_dev(sc->sc_dev, "invalid PCI data" 5049ad1548eSmacallan " signature (%08x)\n", tmp); 5059ad1548eSmacallan rc = EINVAL; 5069ad1548eSmacallan } else { 5079ad1548eSmacallan DPRINTF((" invalid PCI data signature %08x\n", 5089ad1548eSmacallan tmp)); 5099ad1548eSmacallan continue; 5109ad1548eSmacallan } 5119ad1548eSmacallan } 5129ad1548eSmacallan 5139ad1548eSmacallan tmp = bus_space_read_1(pa->pa_memt, romh, suboffs + 0x14); 5149ad1548eSmacallan summitfb_disable_rom_internal(spc); 5159ad1548eSmacallan DPRINTF((" code %02x", tmp)); 5169ad1548eSmacallan 5179ad1548eSmacallan switch (tmp) { 5189ad1548eSmacallan #ifdef __hppa__ 5199ad1548eSmacallan case 0x10: 5209ad1548eSmacallan if (selected == (bus_addr_t)-1) 5219ad1548eSmacallan selected = offs; 5229ad1548eSmacallan break; 5239ad1548eSmacallan #endif 5249ad1548eSmacallan #ifdef __i386__ 5259ad1548eSmacallan case 0x00: 5269ad1548eSmacallan if (selected == (bus_addr_t)-1) 5279ad1548eSmacallan selected = offs; 5289ad1548eSmacallan break; 5299ad1548eSmacallan #endif 5309ad1548eSmacallan default: 5319ad1548eSmacallan DPRINTF((" (wrong architecture)")); 5329ad1548eSmacallan break; 5339ad1548eSmacallan } 5349ad1548eSmacallan DPRINTF(("%s\n", selected == offs ? " -> SELECTED" : "")); 5359ad1548eSmacallan } 5369ad1548eSmacallan 5379ad1548eSmacallan if (selected == (bus_addr_t)-1) { 5389ad1548eSmacallan if (rc == 0) { 5399ad1548eSmacallan aprint_error_dev(sc->sc_dev, "found no ROM with " 5409ad1548eSmacallan "correct microcode architecture\n"); 5419ad1548eSmacallan rc = ENOEXEC; 5429ad1548eSmacallan } 5439ad1548eSmacallan goto fail; 5449ad1548eSmacallan } 5459ad1548eSmacallan 5469ad1548eSmacallan /* 5479ad1548eSmacallan * Read the STI region BAR assignments. 5489ad1548eSmacallan */ 5499ad1548eSmacallan 5509ad1548eSmacallan summitfb_enable_rom_internal(spc); 55138fdde35Sskrll offs = selected + bus_space_read_2(pa->pa_memt, romh, selected + 0x0e); 5529ad1548eSmacallan for (i = 0; i < STI_REGION_MAX; i++) { 5539ad1548eSmacallan rc = summitfb_readbar(sc, pa, i, 5549ad1548eSmacallan bus_space_read_1(pa->pa_memt, romh, offs + i)); 5559ad1548eSmacallan if (rc != 0) 5569ad1548eSmacallan goto fail; 5579ad1548eSmacallan } 5589ad1548eSmacallan 5599ad1548eSmacallan /* 5609ad1548eSmacallan * Find out where the STI ROM itself lies, and its size. 5619ad1548eSmacallan */ 5629ad1548eSmacallan 5639ad1548eSmacallan offs = selected + 56438fdde35Sskrll bus_space_read_4(pa->pa_memt, romh, selected + 0x08); 56538fdde35Sskrll stiromsize = bus_space_read_4(pa->pa_memt, romh, offs + 0x18); 5669ad1548eSmacallan stiromsize = le32toh(stiromsize); 5679ad1548eSmacallan summitfb_disable_rom_internal(spc); 5689ad1548eSmacallan 5699ad1548eSmacallan /* 5709ad1548eSmacallan * Replace our mapping with a smaller mapping of only the area 5719ad1548eSmacallan * we are interested in. 5729ad1548eSmacallan */ 5739ad1548eSmacallan 5749ad1548eSmacallan DPRINTF(("remapping rom @ %lx for %lx\n", 5759ad1548eSmacallan (long)(PCI_MAPREG_ROM_ADDR(address) + offs), (long)stiromsize)); 5769ad1548eSmacallan bus_space_unmap(pa->pa_memt, romh, romsize); 5779ad1548eSmacallan rc = bus_space_map(pa->pa_memt, PCI_MAPREG_ROM_ADDR(address) + offs, 5789ad1548eSmacallan stiromsize, 0, &spc->sc_romh); 5799ad1548eSmacallan if (rc != 0) { 5809ad1548eSmacallan aprint_error_dev(sc->sc_dev, "can't map STI ROM (%d)\n", 5819ad1548eSmacallan rc); 5829ad1548eSmacallan goto fail2; 5839ad1548eSmacallan } 5849ad1548eSmacallan summitfb_disable_rom_internal(spc); 5859ad1548eSmacallan sc->sc_flags &= ~STI_ROM_ENABLED; 5869ad1548eSmacallan 5879ad1548eSmacallan return 0; 5889ad1548eSmacallan 5899ad1548eSmacallan fail: 5909ad1548eSmacallan bus_space_unmap(pa->pa_memt, romh, romsize); 5919ad1548eSmacallan fail2: 5929ad1548eSmacallan summitfb_disable_rom_internal(spc); 5939ad1548eSmacallan 5949ad1548eSmacallan return rc; 5959ad1548eSmacallan } 5969ad1548eSmacallan 5979ad1548eSmacallan /* 5989ad1548eSmacallan * Decode a BAR register. 5999ad1548eSmacallan */ 6009ad1548eSmacallan int 601c84cdbddSriastradh summitfb_readbar(struct sti_softc *sc, struct pci_attach_args *pa, 602c84cdbddSriastradh u_int region, int bar) 6039ad1548eSmacallan { 6049ad1548eSmacallan bus_addr_t addr; 6059ad1548eSmacallan bus_size_t size; 6069ad1548eSmacallan uint32_t cf; 6079ad1548eSmacallan int rc; 6089ad1548eSmacallan 6099ad1548eSmacallan if (bar == 0) { 6109ad1548eSmacallan sc->bases[region] = 0; 611c84cdbddSriastradh return 0; 6129ad1548eSmacallan } 6139ad1548eSmacallan 6149ad1548eSmacallan #ifdef DIAGNOSTIC 6159ad1548eSmacallan if (bar < PCI_MAPREG_START || bar > PCI_MAPREG_PPB_END) { 6169ad1548eSmacallan summitfb_disable_rom(sc); 6179ad1548eSmacallan printf("%s: unexpected bar %02x for region %d\n", 6189ad1548eSmacallan device_xname(sc->sc_dev), bar, region); 6199ad1548eSmacallan summitfb_enable_rom(sc); 6209ad1548eSmacallan } 6219ad1548eSmacallan #endif 6229ad1548eSmacallan 6239ad1548eSmacallan cf = pci_conf_read(pa->pa_pc, pa->pa_tag, bar); 6249ad1548eSmacallan 6259ad1548eSmacallan rc = pci_mapreg_info(pa->pa_pc, pa->pa_tag, bar, PCI_MAPREG_TYPE(cf), 6269ad1548eSmacallan &addr, &size, NULL); 6279ad1548eSmacallan 6289ad1548eSmacallan if (rc != 0) { 6299ad1548eSmacallan summitfb_disable_rom(sc); 630c84cdbddSriastradh aprint_error_dev(sc->sc_dev, "invalid bar %02x" 631c84cdbddSriastradh " for region %d\n", 6329ad1548eSmacallan bar, region); 6339ad1548eSmacallan summitfb_enable_rom(sc); 634c84cdbddSriastradh return rc; 6359ad1548eSmacallan } 6369ad1548eSmacallan 6379ad1548eSmacallan sc->bases[region] = addr; 638c84cdbddSriastradh return 0; 6399ad1548eSmacallan } 6409ad1548eSmacallan 6419ad1548eSmacallan /* 6429ad1548eSmacallan * Enable PCI ROM. 6439ad1548eSmacallan */ 6449ad1548eSmacallan void 6459ad1548eSmacallan summitfb_enable_rom_internal(struct summitfb_softc *spc) 6469ad1548eSmacallan { 6479ad1548eSmacallan pcireg_t address; 6489ad1548eSmacallan 6499ad1548eSmacallan KASSERT(spc != NULL); 6509ad1548eSmacallan 6519ad1548eSmacallan address = pci_conf_read(spc->sc_pc, spc->sc_tag, PCI_MAPREG_ROM); 6529ad1548eSmacallan address |= PCI_MAPREG_ROM_ENABLE; 6539ad1548eSmacallan pci_conf_write(spc->sc_pc, spc->sc_tag, PCI_MAPREG_ROM, address); 6549ad1548eSmacallan } 6559ad1548eSmacallan 6569ad1548eSmacallan void 6579ad1548eSmacallan summitfb_enable_rom(struct sti_softc *sc) 6589ad1548eSmacallan { 6599ad1548eSmacallan struct summitfb_softc *spc = device_private(sc->sc_dev); 6609ad1548eSmacallan 6619ad1548eSmacallan if (!ISSET(sc->sc_flags, STI_ROM_ENABLED)) { 6629ad1548eSmacallan summitfb_enable_rom_internal(spc); 6639ad1548eSmacallan } 6649ad1548eSmacallan SET(sc->sc_flags, STI_ROM_ENABLED); 6659ad1548eSmacallan } 6669ad1548eSmacallan 6679ad1548eSmacallan /* 6689ad1548eSmacallan * Disable PCI ROM. 6699ad1548eSmacallan */ 6709ad1548eSmacallan void 6719ad1548eSmacallan summitfb_disable_rom_internal(struct summitfb_softc *spc) 6729ad1548eSmacallan { 6739ad1548eSmacallan pcireg_t address; 6749ad1548eSmacallan 6759ad1548eSmacallan KASSERT(spc != NULL); 6769ad1548eSmacallan 6779ad1548eSmacallan address = pci_conf_read(spc->sc_pc, spc->sc_tag, PCI_MAPREG_ROM); 6789ad1548eSmacallan address &= ~PCI_MAPREG_ROM_ENABLE; 6799ad1548eSmacallan pci_conf_write(spc->sc_pc, spc->sc_tag, PCI_MAPREG_ROM, address); 6809ad1548eSmacallan } 6819ad1548eSmacallan 6829ad1548eSmacallan void 6839ad1548eSmacallan summitfb_disable_rom(struct sti_softc *sc) 6849ad1548eSmacallan { 6859ad1548eSmacallan struct summitfb_softc *spc = device_private(sc->sc_dev); 6869ad1548eSmacallan 6879ad1548eSmacallan if (ISSET(sc->sc_flags, STI_ROM_ENABLED)) { 6889ad1548eSmacallan summitfb_disable_rom_internal(spc); 6899ad1548eSmacallan } 6909ad1548eSmacallan CLR(sc->sc_flags, STI_ROM_ENABLED); 6919ad1548eSmacallan } 6929ad1548eSmacallan 6939ad1548eSmacallan static inline void 6949ad1548eSmacallan summitfb_wait(struct summitfb_softc *sc) 6959ad1548eSmacallan { 6969ad1548eSmacallan 697c84cdbddSriastradh while (summitfb_read4(sc, VISFX_STATUS) != 0) 698c84cdbddSriastradh continue; 6999ad1548eSmacallan } 7009ad1548eSmacallan 7019ad1548eSmacallan static inline void 7029ad1548eSmacallan summitfb_setup_fb(struct summitfb_softc *sc) 7039ad1548eSmacallan { 704c84cdbddSriastradh 7051558ebccSmacallan summitfb_wait(sc); 706bc72ec74Smacallan if (sc->sc_mode == WSDISPLAYIO_MODE_EMUL) { 707e74ab3bcSmacallan summitfb_write_mode(sc, VISFX_WRITE_MODE_PLAIN); 7085ac86b16Smacallan summitfb_read_mode(sc, VISFX_WRITE_MODE_PLAIN); 709bc72ec74Smacallan summitfb_write4(sc, VISFX_APERTURE_ACCESS, VISFX_DEPTH_8); 710*86973f0aSmacallan /* make overlay opaque */ 711*86973f0aSmacallan summitfb_write4(sc, VISFX_OTR, OTR_T | OTR_L1 | OTR_L0); 712bc72ec74Smacallan } else { 713bc72ec74Smacallan summitfb_write_mode(sc, OTC01 | BIN8F | BUFFL); 7145ac86b16Smacallan summitfb_read_mode(sc, OTC01 | BIN8F | BUFFL); 715bc72ec74Smacallan summitfb_write4(sc, VISFX_APERTURE_ACCESS, VISFX_DEPTH_32); 716*86973f0aSmacallan /* make overlay transparent */ 717*86973f0aSmacallan summitfb_write4(sc, VISFX_OTR, OTR_A); 718bc72ec74Smacallan } 7191558ebccSmacallan summitfb_write4(sc, VISFX_IBO, RopSrc); 7209ad1548eSmacallan } 7219ad1548eSmacallan 7229ad1548eSmacallan void 7239ad1548eSmacallan summitfb_setup(struct summitfb_softc *sc) 7249ad1548eSmacallan { 725bc72ec74Smacallan int i; 7269ad1548eSmacallan 7279ad1548eSmacallan sc->sc_hot_x = 0; 7289ad1548eSmacallan sc->sc_hot_y = 0; 7299ad1548eSmacallan sc->sc_enabled = 0; 7309ad1548eSmacallan sc->sc_video_on = 1; 7319ad1548eSmacallan 7321558ebccSmacallan summitfb_wait(sc); 7339ad1548eSmacallan #if 1 73494d4c668Smacallan /* these control byte swapping */ 73594d4c668Smacallan summitfb_write4(sc, 0xb08044, 0x1b); /* MFU_BSCTD */ 73694d4c668Smacallan summitfb_write4(sc, 0xb08048, 0x1b); /* MFU_BSCCTL */ 73712e4281eSmacallan 73894d4c668Smacallan summitfb_write4(sc, 0x920860, 0xe4); /* FBC_RBS */ 739d2ce6272Smacallan summitfb_write4(sc, 0x921114, 0); /* CPE, clip plane enable */ 74094d4c668Smacallan summitfb_write4(sc, 0x9211d8, 0); /* FCDA */ 74112e4281eSmacallan 74294d4c668Smacallan summitfb_write4(sc, 0xa00818, 0); /* WORG window origin */ 74394d4c668Smacallan summitfb_write4(sc, 0xa0081c, 0); /* FBS front buffer select*/ 74494d4c668Smacallan summitfb_write4(sc, 0xa00850, 0); /* MISC_CTL */ 74594d4c668Smacallan summitfb_write4(sc, 0xa0086c, 0); /* WCE window clipping enable */ 7469ad1548eSmacallan #endif 74794d4c668Smacallan /* initialize drawiing engine */ 748e74ab3bcSmacallan summitfb_wait(sc); 7491558ebccSmacallan summitfb_write4(sc, VISFX_CONTROL, 0); // clear WFC 75012e4281eSmacallan summitfb_write4(sc, VISFX_APERTURE_ACCESS, VISFX_DEPTH_8); 7519ad1548eSmacallan summitfb_write4(sc, VISFX_PIXEL_MASK, 0xffffffff); 7529ad1548eSmacallan summitfb_write4(sc, VISFX_PLANE_MASK, 0xffffffff); 7531558ebccSmacallan summitfb_write4(sc, VISFX_FOE, FOE_BLEND_ROP); 7541558ebccSmacallan summitfb_write4(sc, VISFX_IBO, RopSrc); 755e74ab3bcSmacallan summitfb_write_mode(sc, VISFX_WRITE_MODE_PLAIN); 756d2ce6272Smacallan summitfb_read_mode(sc, OTC04 | BIN8I | BUFovl); 75712e4281eSmacallan summitfb_write4(sc, VISFX_CLIP_TL, 0); 75812e4281eSmacallan summitfb_write4(sc, VISFX_CLIP_WH, 7591558ebccSmacallan ((sc->sc_scr.fbwidth) << 16) | (sc->sc_scr.fbheight)); 76073069606Smacallan /* turn off the cursor sprite */ 76173069606Smacallan summitfb_write4(sc, VISFX_CURSOR_POS, 0); 762*86973f0aSmacallan /* disable throttling by moving the throttle window way off screen */ 763*86973f0aSmacallan summitfb_write4(sc, VISFX_TCR, 0x10001000); 76494d4c668Smacallan 76594d4c668Smacallan /* make sure the overlay is opaque */ 76694d4c668Smacallan summitfb_write4(sc, VISFX_OTR, OTR_T | OTR_L1 | OTR_L0); 76794d4c668Smacallan 76894d4c668Smacallan /* 7695ac86b16Smacallan * initialize XLUT, I mean attribute table 7705ac86b16Smacallan * set all to 24bit, CFS1 77194d4c668Smacallan */ 77294d4c668Smacallan for (i = 0; i < 16; i++) 7735ac86b16Smacallan summitfb_write4(sc, VISFX_IAA(i), IAA_8F | IAA_CFS1); 7745ac86b16Smacallan /* RGB8, no LUT */ 77594d4c668Smacallan summitfb_write4(sc, VISFX_CFS(1), CFS_8F | CFS_BYPASS); 77694d4c668Smacallan /* overlay is 8bit, uses LUT 0 */ 77794d4c668Smacallan summitfb_write4(sc, VISFX_CFS(16), CFS_8I | CFS_LUT0); 77894d4c668Smacallan summitfb_write4(sc, VISFX_CFS(17), CFS_8I | CFS_LUT0); 77994d4c668Smacallan 78094d4c668Smacallan /* zero the attribute plane */ 78194d4c668Smacallan summitfb_write_mode(sc, OTC04 | BINapln); 782d2ce6272Smacallan summitfb_wait_fifo(sc, 12); 78394d4c668Smacallan summitfb_write4(sc, VISFX_PLANE_MASK, 0xff); 7845ac86b16Smacallan summitfb_write4(sc, VISFX_IBO, 0); /* GXclear */ 785d2ce6272Smacallan summitfb_write4(sc, VISFX_FG_COLOUR, 0); 78694d4c668Smacallan summitfb_write4(sc, VISFX_START, 0); 78794d4c668Smacallan summitfb_write4(sc, VISFX_SIZE, (sc->sc_width << 16) | sc->sc_height); 78894d4c668Smacallan summitfb_wait(sc); 78994d4c668Smacallan summitfb_write4(sc, VISFX_PLANE_MASK, 0xffffffff); 79094d4c668Smacallan 79194d4c668Smacallan /* turn off force attr so the above takes effect */ 79294d4c668Smacallan summitfb_write4(sc, VISFX_FATTR, 0); 79394d4c668Smacallan 7949ad1548eSmacallan summitfb_setup_fb(sc); 7959ad1548eSmacallan } 7969ad1548eSmacallan 7979ad1548eSmacallan static int 7989ad1548eSmacallan summitfb_ioctl(void *v, void *vs, u_long cmd, void *data, int flag, 7999ad1548eSmacallan struct lwp *l) 8009ad1548eSmacallan { 8019ad1548eSmacallan struct vcons_data *vd = v; 8029ad1548eSmacallan struct summitfb_softc *sc = vd->cookie; 8039ad1548eSmacallan struct wsdisplay_fbinfo *wdf; 8049ad1548eSmacallan struct vcons_screen *ms = vd->active; 8059ad1548eSmacallan 8069ad1548eSmacallan switch (cmd) { 8079ad1548eSmacallan case WSDISPLAYIO_GTYPE: 8089ad1548eSmacallan *(u_int *)data = WSDISPLAY_TYPE_STI; 8099ad1548eSmacallan return 0; 8109ad1548eSmacallan 8119ad1548eSmacallan case GCID: 812f7327fc7Smacallan *(u_int *)data = sc->sc_scr.scr_rom->rom_dd.dd_grid[0]; 8139ad1548eSmacallan return 0; 8149ad1548eSmacallan 8159ad1548eSmacallan /* PCI config read/write passthrough. */ 8169ad1548eSmacallan case PCI_IOC_CFGREAD: 8179ad1548eSmacallan case PCI_IOC_CFGWRITE: 8189ad1548eSmacallan return pci_devioctl(sc->sc_pc, sc->sc_tag, 8199ad1548eSmacallan cmd, data, flag, l); 8209ad1548eSmacallan 8219ad1548eSmacallan case WSDISPLAYIO_GET_BUSID: 8229ad1548eSmacallan return wsdisplayio_busid_pci(sc->sc_dev, sc->sc_pc, 8239ad1548eSmacallan sc->sc_tag, data); 8249ad1548eSmacallan 8259ad1548eSmacallan case WSDISPLAYIO_GINFO: 8269ad1548eSmacallan if (ms == NULL) 8279ad1548eSmacallan return ENODEV; 828c84cdbddSriastradh wdf = data; 8299ad1548eSmacallan wdf->height = ms->scr_ri.ri_height; 8309ad1548eSmacallan wdf->width = ms->scr_ri.ri_width; 8319ad1548eSmacallan wdf->depth = ms->scr_ri.ri_depth; 8329ad1548eSmacallan wdf->cmsize = 256; 8339ad1548eSmacallan return 0; 8349ad1548eSmacallan 8359ad1548eSmacallan case WSDISPLAYIO_GETCMAP: 8369ad1548eSmacallan return summitfb_getcmap(sc, 8379ad1548eSmacallan (struct wsdisplay_cmap *)data); 8389ad1548eSmacallan 8399ad1548eSmacallan case WSDISPLAYIO_PUTCMAP: 8409ad1548eSmacallan return summitfb_putcmap(sc, 8419ad1548eSmacallan (struct wsdisplay_cmap *)data); 8429ad1548eSmacallan 8439ad1548eSmacallan case WSDISPLAYIO_LINEBYTES: 8449ad1548eSmacallan *(u_int *)data = 2048; 8459ad1548eSmacallan return 0; 8469ad1548eSmacallan 8479ad1548eSmacallan case WSDISPLAYIO_SMODE: { 8489ad1548eSmacallan int new_mode = *(int *)data; 849c84cdbddSriastradh 8509ad1548eSmacallan if (new_mode != sc->sc_mode) { 8519ad1548eSmacallan sc->sc_mode = new_mode; 8529ad1548eSmacallan if(new_mode == WSDISPLAYIO_MODE_EMUL) { 8539ad1548eSmacallan summitfb_setup(sc); 8549ad1548eSmacallan summitfb_restore_palette(sc); 85512e4281eSmacallan glyphcache_wipe(&sc->sc_gc); 8567bbb04c1Smacallan summitfb_loadfont(sc); 8579ad1548eSmacallan summitfb_rectfill(sc, 0, 0, sc->sc_width, 8589ad1548eSmacallan sc->sc_height, ms->scr_ri.ri_devcmap[ 8599ad1548eSmacallan (ms->scr_defattr >> 16) & 0xff]); 8609ad1548eSmacallan vcons_redraw_screen(ms); 8619ad1548eSmacallan summitfb_set_video(sc, 1); 862d2ce6272Smacallan } else 863d2ce6272Smacallan summitfb_clearfb(sc); 864e74ab3bcSmacallan summitfb_setup_fb(sc); 8659ad1548eSmacallan } 8669ad1548eSmacallan return 0; 867c84cdbddSriastradh } 8689ad1548eSmacallan 869c84cdbddSriastradh case WSDISPLAYIO_GET_FBINFO: { 8709ad1548eSmacallan struct wsdisplayio_fbinfo *fbi = data; 8719ad1548eSmacallan int ret; 8729ad1548eSmacallan 8739ad1548eSmacallan ret = wsdisplayio_get_fbinfo(&ms->scr_ri, fbi); 874bc72ec74Smacallan //fbi->fbi_fbsize = sc->sc_height * 2048; 875bc72ec74Smacallan fbi->fbi_stride = 8192; 876bc72ec74Smacallan fbi->fbi_bitsperpixel = 32; 877bc72ec74Smacallan fbi->fbi_pixeltype = WSFB_RGB; 878bc72ec74Smacallan fbi->fbi_subtype.fbi_rgbmasks.red_offset = 16; 879bc72ec74Smacallan fbi->fbi_subtype.fbi_rgbmasks.red_size = 8; 880bc72ec74Smacallan fbi->fbi_subtype.fbi_rgbmasks.green_offset = 8; 881bc72ec74Smacallan fbi->fbi_subtype.fbi_rgbmasks.green_size = 8; 882bc72ec74Smacallan fbi->fbi_subtype.fbi_rgbmasks.blue_offset = 0; 883bc72ec74Smacallan fbi->fbi_subtype.fbi_rgbmasks.blue_size = 8; 884bc72ec74Smacallan fbi->fbi_subtype.fbi_rgbmasks.alpha_size = 0; 885bc72ec74Smacallan fbi->fbi_fbsize = sc->sc_scr.fbheight * 8192; 8869ad1548eSmacallan return ret; 8879ad1548eSmacallan } 8889ad1548eSmacallan 889c84cdbddSriastradh case WSDISPLAYIO_GCURPOS: { 890c84cdbddSriastradh struct wsdisplay_curpos *cp = data; 8919ad1548eSmacallan 8929ad1548eSmacallan cp->x = sc->sc_cursor_x; 8939ad1548eSmacallan cp->y = sc->sc_cursor_y; 8949ad1548eSmacallan return 0; 895c84cdbddSriastradh } 8969ad1548eSmacallan 897c84cdbddSriastradh case WSDISPLAYIO_SCURPOS: { 898c84cdbddSriastradh struct wsdisplay_curpos *cp = data; 8999ad1548eSmacallan 9009ad1548eSmacallan summitfb_move_cursor(sc, cp->x, cp->y); 9019ad1548eSmacallan return 0; 902c84cdbddSriastradh } 9039ad1548eSmacallan 904c84cdbddSriastradh case WSDISPLAYIO_GCURMAX: { 905c84cdbddSriastradh struct wsdisplay_curpos *cp = data; 9069ad1548eSmacallan 9079ad1548eSmacallan cp->x = 64; 9089ad1548eSmacallan cp->y = 64; 9099ad1548eSmacallan return 0; 910c84cdbddSriastradh } 9119ad1548eSmacallan 912c84cdbddSriastradh case WSDISPLAYIO_SCURSOR: { 913c84cdbddSriastradh struct wsdisplay_cursor *cursor = data; 9149ad1548eSmacallan 9159ad1548eSmacallan return summitfb_do_cursor(sc, cursor); 9169ad1548eSmacallan } 9179ad1548eSmacallan 9189ad1548eSmacallan case WSDISPLAYIO_SVIDEO: 9199ad1548eSmacallan summitfb_set_video(sc, *(int *)data); 9209ad1548eSmacallan return 0; 9219ad1548eSmacallan case WSDISPLAYIO_GVIDEO: 9229ad1548eSmacallan *(u_int *)data = sc->sc_video_on ? 9239ad1548eSmacallan WSDISPLAYIO_VIDEO_ON : WSDISPLAYIO_VIDEO_OFF; 9249ad1548eSmacallan return 0; 9259ad1548eSmacallan } 9269ad1548eSmacallan return EPASSTHROUGH; 9279ad1548eSmacallan } 9289ad1548eSmacallan 9299ad1548eSmacallan static paddr_t 9309ad1548eSmacallan summitfb_mmap(void *v, void *vs, off_t offset, int prot) 9319ad1548eSmacallan { 9329ad1548eSmacallan struct vcons_data *vd = v; 9339ad1548eSmacallan struct summitfb_softc *sc = vd->cookie; 9349ad1548eSmacallan struct sti_rom *rom = sc->sc_base.sc_rom; 9359ad1548eSmacallan paddr_t pa = -1; 9369ad1548eSmacallan 9379ad1548eSmacallan if (sc->sc_mode == WSDISPLAYIO_MODE_EMUL) 9389ad1548eSmacallan return -1; 9399ad1548eSmacallan 940d7adffb3Smacallan if (offset >= 0 && offset < 0x01000000) { 9419ad1548eSmacallan /* framebuffer */ 9429ad1548eSmacallan pa = bus_space_mmap(rom->memt, sc->sc_scr.fbaddr, offset, 9439ad1548eSmacallan prot, BUS_SPACE_MAP_LINEAR); 9446fcfb3adSmacallan } else if (offset >= 0x80000000 && offset < 0x81000000) { 9459ad1548eSmacallan /* blitter registers etc. */ 946f7327fc7Smacallan pa = bus_space_mmap(rom->memt, rom->regh[0], 9479ad1548eSmacallan offset - 0x80000000, prot, BUS_SPACE_MAP_LINEAR); 9489ad1548eSmacallan } 9499ad1548eSmacallan 9509ad1548eSmacallan return pa; 9519ad1548eSmacallan } 9529ad1548eSmacallan 9539ad1548eSmacallan static void 9549ad1548eSmacallan summitfb_init_screen(void *cookie, struct vcons_screen *scr, 9559ad1548eSmacallan int existing, long *defattr) 9569ad1548eSmacallan { 9579ad1548eSmacallan struct summitfb_softc *sc = cookie; 9589ad1548eSmacallan struct rasops_info *ri = &scr->scr_ri; 9599ad1548eSmacallan 9609ad1548eSmacallan ri->ri_depth = 8; 9619ad1548eSmacallan ri->ri_width = sc->sc_width; 9629ad1548eSmacallan ri->ri_height = sc->sc_height; 9639ad1548eSmacallan ri->ri_stride = 2048; 9647bbb04c1Smacallan ri->ri_flg = RI_CENTER | RI_8BIT_IS_RGB 9657bbb04c1Smacallan | RI_ENABLE_ALPHA | RI_PREFER_ALPHA 9667bbb04c1Smacallan ; 9679ad1548eSmacallan 9689ad1548eSmacallan ri->ri_bits = (void *)sc->sc_scr.fbaddr; 9699ad1548eSmacallan rasops_init(ri, 0, 0); 9709ad1548eSmacallan ri->ri_caps = WSSCREEN_WSCOLORS | WSSCREEN_HILIT | WSSCREEN_UNDERLINE | 9719ad1548eSmacallan WSSCREEN_RESIZE; 9725ac86b16Smacallan scr->scr_flags |= VCONS_LOADFONT; 9739ad1548eSmacallan 9749ad1548eSmacallan rasops_reconfig(ri, sc->sc_height / ri->ri_font->fontheight, 9759ad1548eSmacallan sc->sc_width / ri->ri_font->fontwidth); 9769ad1548eSmacallan 9779ad1548eSmacallan ri->ri_hw = scr; 9781dd4359cSmacallan 9799ad1548eSmacallan ri->ri_ops.copyrows = summitfb_copyrows; 9809ad1548eSmacallan ri->ri_ops.copycols = summitfb_copycols; 9819ad1548eSmacallan ri->ri_ops.eraserows = summitfb_eraserows; 9829ad1548eSmacallan ri->ri_ops.erasecols = summitfb_erasecols; 9831558ebccSmacallan ri->ri_ops.cursor = summitfb_cursor; 9845ac86b16Smacallan sc->sc_font = NULL; 9859ad1548eSmacallan if (FONT_IS_ALPHA(ri->ri_font)) { 9869ad1548eSmacallan ri->ri_ops.putchar = summitfb_putchar_aa; 9879ad1548eSmacallan } else 9887bbb04c1Smacallan { 9897bbb04c1Smacallan int fbwidth = (sc->sc_width + 511) & ~511; 9907bbb04c1Smacallan int fcols = (fbwidth - sc->sc_width - 2) / ri->ri_font->fontwidth; 9917bbb04c1Smacallan int frows = sc->sc_height / ri->ri_font->fontheight; 9927bbb04c1Smacallan sc->sc_font_start = sc->sc_width + 2; 9937bbb04c1Smacallan if ((fcols * frows) >= ri->ri_font->numchars) { 9947bbb04c1Smacallan /* ok, we can do this */ 9957bbb04c1Smacallan sc->sc_cols = fcols; 9967bbb04c1Smacallan sc->sc_font = ri->ri_font; 9977bbb04c1Smacallan summitfb_loadfont(sc); 9987bbb04c1Smacallan ri->ri_ops.putchar = summitfb_putchar_fast; 9997bbb04c1Smacallan } else 10009ad1548eSmacallan ri->ri_ops.putchar = summitfb_putchar; 10019ad1548eSmacallan } 10027bbb04c1Smacallan } 10039ad1548eSmacallan 10049ad1548eSmacallan static int 10059ad1548eSmacallan summitfb_putcmap(struct summitfb_softc *sc, struct wsdisplay_cmap *cm) 10069ad1548eSmacallan { 10079ad1548eSmacallan u_char *r, *g, *b; 10089ad1548eSmacallan u_int index = cm->index; 10099ad1548eSmacallan u_int count = cm->count; 10109ad1548eSmacallan int i, error; 10119ad1548eSmacallan u_char rbuf[256], gbuf[256], bbuf[256]; 10129ad1548eSmacallan 10139ad1548eSmacallan if (cm->index >= 256 || cm->count > 256 || 10149ad1548eSmacallan (cm->index + cm->count) > 256) 10159ad1548eSmacallan return EINVAL; 10169ad1548eSmacallan error = copyin(cm->red, &rbuf[index], count); 10179ad1548eSmacallan if (error) 10189ad1548eSmacallan return error; 10199ad1548eSmacallan error = copyin(cm->green, &gbuf[index], count); 10209ad1548eSmacallan if (error) 10219ad1548eSmacallan return error; 10229ad1548eSmacallan error = copyin(cm->blue, &bbuf[index], count); 10239ad1548eSmacallan if (error) 10249ad1548eSmacallan return error; 10259ad1548eSmacallan 10269ad1548eSmacallan memcpy(&sc->sc_cmap_red[index], &rbuf[index], count); 10279ad1548eSmacallan memcpy(&sc->sc_cmap_green[index], &gbuf[index], count); 10289ad1548eSmacallan memcpy(&sc->sc_cmap_blue[index], &bbuf[index], count); 10299ad1548eSmacallan 10309ad1548eSmacallan r = &sc->sc_cmap_red[index]; 10319ad1548eSmacallan g = &sc->sc_cmap_green[index]; 10329ad1548eSmacallan b = &sc->sc_cmap_blue[index]; 10339ad1548eSmacallan 10349ad1548eSmacallan for (i = 0; i < count; i++) { 10359ad1548eSmacallan summitfb_putpalreg(sc, index, *r, *g, *b); 10369ad1548eSmacallan index++; 10379ad1548eSmacallan r++, g++, b++; 10389ad1548eSmacallan } 10399ad1548eSmacallan return 0; 10409ad1548eSmacallan } 10419ad1548eSmacallan 10429ad1548eSmacallan static int 10439ad1548eSmacallan summitfb_getcmap(struct summitfb_softc *sc, struct wsdisplay_cmap *cm) 10449ad1548eSmacallan { 10459ad1548eSmacallan u_int index = cm->index; 10469ad1548eSmacallan u_int count = cm->count; 10479ad1548eSmacallan int error; 10489ad1548eSmacallan 10499ad1548eSmacallan if (index >= 255 || count > 256 || index + count > 256) 10509ad1548eSmacallan return EINVAL; 10519ad1548eSmacallan 10529ad1548eSmacallan error = copyout(&sc->sc_cmap_red[index], cm->red, count); 10539ad1548eSmacallan if (error) 10549ad1548eSmacallan return error; 10559ad1548eSmacallan error = copyout(&sc->sc_cmap_green[index], cm->green, count); 10569ad1548eSmacallan if (error) 10579ad1548eSmacallan return error; 10589ad1548eSmacallan error = copyout(&sc->sc_cmap_blue[index], cm->blue, count); 10599ad1548eSmacallan if (error) 10609ad1548eSmacallan return error; 10619ad1548eSmacallan 10629ad1548eSmacallan return 0; 10639ad1548eSmacallan } 10649ad1548eSmacallan 10659ad1548eSmacallan static void 10669ad1548eSmacallan summitfb_restore_palette(struct summitfb_softc *sc) 10679ad1548eSmacallan { 10689ad1548eSmacallan uint8_t cmap[768]; 10699ad1548eSmacallan int i, j; 10709ad1548eSmacallan 10719ad1548eSmacallan j = 0; 10729ad1548eSmacallan rasops_get_cmap(&sc->sc_console_screen.scr_ri, cmap, sizeof(cmap)); 10739ad1548eSmacallan for (i = 0; i < 256; i++) { 10749ad1548eSmacallan sc->sc_cmap_red[i] = cmap[j]; 10759ad1548eSmacallan sc->sc_cmap_green[i] = cmap[j + 1]; 10769ad1548eSmacallan sc->sc_cmap_blue[i] = cmap[j + 2]; 10779ad1548eSmacallan summitfb_putpalreg(sc, i, cmap[j], cmap[j + 1], cmap[j + 2]); 10789ad1548eSmacallan j += 3; 10799ad1548eSmacallan } 1080*86973f0aSmacallan for (i = 0; i < 16; i++) { 1081*86973f0aSmacallan sc->sc_palette[i] = (rasops_cmap[i * 3] << 16) | 1082*86973f0aSmacallan (rasops_cmap[i * 3 + 1] << 8) | 1083*86973f0aSmacallan rasops_cmap[i * 3 + 2]; 1084*86973f0aSmacallan } 1085*86973f0aSmacallan 10869ad1548eSmacallan } 10879ad1548eSmacallan 10889ad1548eSmacallan static int 1089c84cdbddSriastradh summitfb_putpalreg(struct summitfb_softc *sc, uint8_t idx, 1090c84cdbddSriastradh uint8_t r, uint8_t g, uint8_t b) 10919ad1548eSmacallan { 1092c84cdbddSriastradh 1093bc72ec74Smacallan summitfb_write4(sc, VISFX_COLOR_INDEX, idx); 10949ad1548eSmacallan summitfb_write4(sc, VISFX_COLOR_VALUE, (r << 16) | ( g << 8) | b); 10959ad1548eSmacallan summitfb_write4(sc, VISFX_COLOR_MASK, 0xff); 10969ad1548eSmacallan return 0; 10979ad1548eSmacallan } 10989ad1548eSmacallan 10999ad1548eSmacallan static inline void 11009ad1548eSmacallan summitfb_wait_fifo(struct summitfb_softc *sc, uint32_t slots) 11019ad1548eSmacallan { 11029ad1548eSmacallan uint32_t reg; 1103c84cdbddSriastradh 11049ad1548eSmacallan do { 11051558ebccSmacallan reg = summitfb_read4(sc, VISFX_FIFO); 11069ad1548eSmacallan } while (reg < slots); 11079ad1548eSmacallan } 11089ad1548eSmacallan 11099ad1548eSmacallan static void 1110d2ce6272Smacallan summitfb_clearfb(struct summitfb_softc *sc) 1111d2ce6272Smacallan { 1112d2ce6272Smacallan summitfb_write_mode(sc, OTC32 | BIN8F | BUFBL | BUFFL | 0x8c0); 1113d2ce6272Smacallan summitfb_wait_fifo(sc, 10); 1114d2ce6272Smacallan summitfb_write4(sc, VISFX_IBO, RopSrc); 1115d2ce6272Smacallan summitfb_write4(sc, VISFX_FG_COLOUR, 0); 1116d2ce6272Smacallan summitfb_write4(sc, VISFX_START, 0); 1117d2ce6272Smacallan summitfb_write4(sc, VISFX_SIZE, (sc->sc_width << 16) | sc->sc_height); 1118d2ce6272Smacallan } 1119d2ce6272Smacallan 1120d2ce6272Smacallan static void 11219ad1548eSmacallan summitfb_rectfill(struct summitfb_softc *sc, int x, int y, int wi, int he, 11229ad1548eSmacallan uint32_t bg) 11239ad1548eSmacallan { 1124c84cdbddSriastradh 1125e74ab3bcSmacallan summitfb_write_mode(sc, VISFX_WRITE_MODE_FILL); 1126d2ce6272Smacallan summitfb_wait_fifo(sc, 10); 11271558ebccSmacallan summitfb_write4(sc, VISFX_IBO, RopSrc); 11289ad1548eSmacallan summitfb_write4(sc, VISFX_FG_COLOUR, bg); 11299ad1548eSmacallan summitfb_write4(sc, VISFX_START, (x << 16) | y); 11309ad1548eSmacallan summitfb_write4(sc, VISFX_SIZE, (wi << 16) | he); 11319ad1548eSmacallan } 11329ad1548eSmacallan 11339ad1548eSmacallan static void 11349ad1548eSmacallan summitfb_bitblt(void *cookie, int xs, int ys, int xd, int yd, int wi, 11359ad1548eSmacallan int he, int rop) 11369ad1548eSmacallan { 11379ad1548eSmacallan struct summitfb_softc *sc = cookie; 11385ac86b16Smacallan uint32_t read_mode, write_mode; 11391dd4359cSmacallan 11405ac86b16Smacallan read_mode = OTC04 | BIN8I; 11415ac86b16Smacallan write_mode = OTC04 | BIN8I; 11425ac86b16Smacallan if (ys >= sc->sc_height) { 11435ac86b16Smacallan read_mode |= BUFBL; 11445ac86b16Smacallan ys -= sc->sc_height; 11455ac86b16Smacallan } 11465ac86b16Smacallan if (yd >= sc->sc_height) { 11475ac86b16Smacallan write_mode |= BUFBL; 11485ac86b16Smacallan yd -= sc->sc_height; 11495ac86b16Smacallan } 11505ac86b16Smacallan summitfb_write_mode(sc, write_mode); 11515ac86b16Smacallan summitfb_read_mode(sc, read_mode); 1152d2ce6272Smacallan summitfb_wait_fifo(sc, 10); 11531558ebccSmacallan summitfb_write4(sc, VISFX_IBO, rop); 11541dd4359cSmacallan summitfb_write4(sc, VISFX_COPY_SRC, (xs << 16) | ys); 11551dd4359cSmacallan summitfb_write4(sc, VISFX_COPY_WH, (wi << 16) | he); 11561dd4359cSmacallan summitfb_write4(sc, VISFX_COPY_DST, (xd << 16) | yd); 11571dd4359cSmacallan 11589ad1548eSmacallan } 11599ad1548eSmacallan 11609ad1548eSmacallan static void 11619ad1548eSmacallan summitfb_nuke_cursor(struct rasops_info *ri) 11629ad1548eSmacallan { 11639ad1548eSmacallan struct vcons_screen *scr = ri->ri_hw; 11649ad1548eSmacallan struct summitfb_softc *sc = scr->scr_cookie; 11659ad1548eSmacallan int wi, he, x, y; 11669ad1548eSmacallan 11679ad1548eSmacallan if (ri->ri_flg & RI_CURSOR) { 11689ad1548eSmacallan wi = ri->ri_font->fontwidth; 11699ad1548eSmacallan he = ri->ri_font->fontheight; 11709ad1548eSmacallan x = ri->ri_ccol * wi + ri->ri_xorigin; 11719ad1548eSmacallan y = ri->ri_crow * he + ri->ri_yorigin; 11721558ebccSmacallan summitfb_bitblt(sc, x, y, x, y, wi, he, RopInv); 11739ad1548eSmacallan ri->ri_flg &= ~RI_CURSOR; 11749ad1548eSmacallan } 11759ad1548eSmacallan } 11769ad1548eSmacallan 11779ad1548eSmacallan static void 11789ad1548eSmacallan summitfb_cursor(void *cookie, int on, int row, int col) 11799ad1548eSmacallan { 11809ad1548eSmacallan struct rasops_info *ri = cookie; 11819ad1548eSmacallan struct vcons_screen *scr = ri->ri_hw; 11829ad1548eSmacallan struct summitfb_softc *sc = scr->scr_cookie; 11839ad1548eSmacallan int x, y, wi, he; 11849ad1548eSmacallan 11859ad1548eSmacallan wi = ri->ri_font->fontwidth; 11869ad1548eSmacallan he = ri->ri_font->fontheight; 11879ad1548eSmacallan 11889ad1548eSmacallan if (sc->sc_mode == WSDISPLAYIO_MODE_EMUL) { 11899ad1548eSmacallan if (on) { 11909ad1548eSmacallan if (ri->ri_flg & RI_CURSOR) { 11919ad1548eSmacallan summitfb_nuke_cursor(ri); 11929ad1548eSmacallan } 11939ad1548eSmacallan x = col * wi + ri->ri_xorigin; 11949ad1548eSmacallan y = row * he + ri->ri_yorigin; 11959ad1548eSmacallan summitfb_bitblt(sc, x, y, x, y, wi, he, RopInv); 11969ad1548eSmacallan ri->ri_flg |= RI_CURSOR; 11979ad1548eSmacallan } 11989ad1548eSmacallan ri->ri_crow = row; 11999ad1548eSmacallan ri->ri_ccol = col; 1200c84cdbddSriastradh } else { 12019ad1548eSmacallan ri->ri_crow = row; 12029ad1548eSmacallan ri->ri_ccol = col; 12039ad1548eSmacallan ri->ri_flg &= ~RI_CURSOR; 12049ad1548eSmacallan } 12059ad1548eSmacallan 12069ad1548eSmacallan } 12079ad1548eSmacallan 12089ad1548eSmacallan static void 12099ad1548eSmacallan summitfb_putchar(void *cookie, int row, int col, u_int c, long attr) 12109ad1548eSmacallan { 12119ad1548eSmacallan struct rasops_info *ri = cookie; 12129ad1548eSmacallan struct wsdisplay_font *font = PICK_FONT(ri, c); 12139ad1548eSmacallan struct vcons_screen *scr = ri->ri_hw; 12149ad1548eSmacallan struct summitfb_softc *sc = scr->scr_cookie; 12151dd4359cSmacallan void *data; 12167bbb04c1Smacallan int i, x, y, wi, he; 12179ad1548eSmacallan uint32_t bg, fg, mask; 12189ad1548eSmacallan 12199ad1548eSmacallan if (sc->sc_mode != WSDISPLAYIO_MODE_EMUL) 12209ad1548eSmacallan return; 12219ad1548eSmacallan 12229ad1548eSmacallan if (!CHAR_IN_FONT(c, font)) 12239ad1548eSmacallan return; 12249ad1548eSmacallan 12259ad1548eSmacallan if (row == ri->ri_crow && col == ri->ri_ccol) { 12269ad1548eSmacallan ri->ri_flg &= ~RI_CURSOR; 12279ad1548eSmacallan } 12281558ebccSmacallan 12299ad1548eSmacallan wi = font->fontwidth; 12309ad1548eSmacallan he = font->fontheight; 12319ad1548eSmacallan 12329ad1548eSmacallan x = ri->ri_xorigin + col * wi; 12339ad1548eSmacallan y = ri->ri_yorigin + row * he; 12349ad1548eSmacallan 12359ad1548eSmacallan bg = ri->ri_devcmap[(attr >> 16) & 0xf]; 12369ad1548eSmacallan 12379ad1548eSmacallan /* if we're drawing a space we're done here */ 12389ad1548eSmacallan if (c == 0x20) { 12399ad1548eSmacallan summitfb_rectfill(sc, x, y, wi, he, bg); 12409ad1548eSmacallan return; 12419ad1548eSmacallan } 12429ad1548eSmacallan 12439ad1548eSmacallan fg = ri->ri_devcmap[(attr >> 24) & 0x0f]; 12449ad1548eSmacallan 1245e74ab3bcSmacallan summitfb_write_mode(sc, VISFX_WRITE_MODE_EXPAND); 1246d2ce6272Smacallan summitfb_wait_fifo(sc, 12); 12471558ebccSmacallan summitfb_write4(sc, VISFX_IBO, RopSrc); 12481dd4359cSmacallan summitfb_write4(sc, VISFX_FG_COLOUR, fg); 12491dd4359cSmacallan summitfb_write4(sc, VISFX_BG_COLOUR, bg); 12501dd4359cSmacallan mask = 0xffffffff << (32 - wi); 12511dd4359cSmacallan summitfb_write4(sc, VISFX_PIXEL_MASK, mask); 12521dd4359cSmacallan /* not a tpyo, coordinates *are* backwards for this register */ 12531dd4359cSmacallan summitfb_write4(sc, VISFX_VRAM_WRITE_DEST, (y << 16) | x); 12549ad1548eSmacallan 12551dd4359cSmacallan data = WSFONT_GLYPH(c, font); 12561dd4359cSmacallan 12571dd4359cSmacallan if (ri->ri_font->stride == 1) { 12581dd4359cSmacallan uint8_t *data8 = data; 12591dd4359cSmacallan for (i = 0; i < he; i++) { 12601dd4359cSmacallan mask = *data8; 1261530bb1baSriastradh summitfb_write4(sc, VISFX_VRAM_WRITE_DATA_INCRY, 1262530bb1baSriastradh mask << 24); 12631dd4359cSmacallan data8++; 12641dd4359cSmacallan } 12651dd4359cSmacallan } else { 12661dd4359cSmacallan uint16_t *data16 = data; 12671dd4359cSmacallan for (i = 0; i < he; i++) { 12681dd4359cSmacallan mask = *data16; 1269530bb1baSriastradh summitfb_write4(sc, VISFX_VRAM_WRITE_DATA_INCRY, 1270530bb1baSriastradh mask << 16); 12711dd4359cSmacallan data16++; 12721dd4359cSmacallan } 12731dd4359cSmacallan } 12749ad1548eSmacallan } 12759ad1548eSmacallan 12769ad1548eSmacallan static void 12777bbb04c1Smacallan summitfb_loadfont(struct summitfb_softc *sc) 12787bbb04c1Smacallan { 12797bbb04c1Smacallan int i, c, x, y; 12807bbb04c1Smacallan uint8_t *data; 12817bbb04c1Smacallan uint16_t *data16; 12827bbb04c1Smacallan uint32_t mask; 12837bbb04c1Smacallan 12845ac86b16Smacallan if (sc->sc_font == NULL) 12855ac86b16Smacallan return; 12865ac86b16Smacallan 12877bbb04c1Smacallan summitfb_write_mode(sc, VISFX_WRITE_MODE_EXPAND); 1288d2ce6272Smacallan summitfb_wait_fifo(sc, 10); 12891558ebccSmacallan summitfb_write4(sc, VISFX_IBO, RopSrc); 12907bbb04c1Smacallan summitfb_write4(sc, VISFX_FG_COLOUR, 0xffffffff); 12917bbb04c1Smacallan summitfb_write4(sc, VISFX_BG_COLOUR, 0); 12927bbb04c1Smacallan 12937bbb04c1Smacallan mask = 0xffffffff << (32 - sc->sc_font->fontwidth); 12947bbb04c1Smacallan summitfb_write4(sc, VISFX_PIXEL_MASK, mask); 12957bbb04c1Smacallan 12967bbb04c1Smacallan for (c = 0; c < sc->sc_font->numchars; c++) { 12977bbb04c1Smacallan x = sc->sc_font_start + (c % sc->sc_cols) * sc->sc_font->fontwidth; 12987bbb04c1Smacallan y = (c / sc->sc_cols) * sc->sc_font->fontheight; 12997bbb04c1Smacallan data = WSFONT_GLYPH(sc->sc_font->firstchar + c, sc->sc_font); 13007bbb04c1Smacallan summitfb_write4(sc, VISFX_VRAM_WRITE_DEST, (y << 16) | x); 13017bbb04c1Smacallan if (sc->sc_font->stride == 1) { 13027bbb04c1Smacallan for (i = 0; i < sc->sc_font->fontheight; i++) { 13037bbb04c1Smacallan mask = *data; 13047bbb04c1Smacallan summitfb_write4(sc, VISFX_VRAM_WRITE_DATA_INCRY, 13057bbb04c1Smacallan mask << 24); 13067bbb04c1Smacallan data++; 13077bbb04c1Smacallan } 13087bbb04c1Smacallan } else { 13097bbb04c1Smacallan data16 = (uint16_t *)data; 13107bbb04c1Smacallan for (i = 0; i < sc->sc_font->fontheight; i++) { 13117bbb04c1Smacallan mask = *data16; 13127bbb04c1Smacallan summitfb_write4(sc, VISFX_VRAM_WRITE_DATA_INCRY, 13137bbb04c1Smacallan mask << 16); 13147bbb04c1Smacallan data16++; 13157bbb04c1Smacallan } 13167bbb04c1Smacallan } 13177bbb04c1Smacallan } 13187bbb04c1Smacallan } 13197bbb04c1Smacallan 13207bbb04c1Smacallan static void 13217bbb04c1Smacallan summitfb_putchar_fast(void *cookie, int row, int col, u_int c, long attr) 13227bbb04c1Smacallan { 13237bbb04c1Smacallan struct rasops_info *ri = cookie; 13247bbb04c1Smacallan struct wsdisplay_font *font = PICK_FONT(ri, c); 13257bbb04c1Smacallan struct vcons_screen *scr = ri->ri_hw; 13267bbb04c1Smacallan struct summitfb_softc *sc = scr->scr_cookie; 13277bbb04c1Smacallan int i, x, y, wi, he, xs, ys; 13287bbb04c1Smacallan uint32_t bg, fg; 13297bbb04c1Smacallan 13307bbb04c1Smacallan if (sc->sc_mode != WSDISPLAYIO_MODE_EMUL) 13317bbb04c1Smacallan return; 13327bbb04c1Smacallan 13337bbb04c1Smacallan if (!CHAR_IN_FONT(c, font)) 13347bbb04c1Smacallan return; 13357bbb04c1Smacallan 13367bbb04c1Smacallan /* for autogenerated line drawing characters */ 13377bbb04c1Smacallan if (font != sc->sc_font) { 13387bbb04c1Smacallan summitfb_putchar(cookie, row, col, c, attr); 13397bbb04c1Smacallan return; 13407bbb04c1Smacallan } 13417bbb04c1Smacallan 13427bbb04c1Smacallan if (row == ri->ri_crow && col == ri->ri_ccol) { 13437bbb04c1Smacallan ri->ri_flg &= ~RI_CURSOR; 13447bbb04c1Smacallan } 13451558ebccSmacallan 13467bbb04c1Smacallan wi = font->fontwidth; 13477bbb04c1Smacallan he = font->fontheight; 13487bbb04c1Smacallan 13497bbb04c1Smacallan x = ri->ri_xorigin + col * wi; 13507bbb04c1Smacallan y = ri->ri_yorigin + row * he; 13517bbb04c1Smacallan 13527bbb04c1Smacallan bg = ri->ri_devcmap[(attr >> 16) & 0xf]; 13537bbb04c1Smacallan 13547bbb04c1Smacallan /* if we're drawing a space we're done here */ 13557bbb04c1Smacallan if (c == 0x20) { 13567bbb04c1Smacallan summitfb_rectfill(sc, x, y, wi, he, bg); 13577bbb04c1Smacallan return; 13587bbb04c1Smacallan } 13597bbb04c1Smacallan 13607bbb04c1Smacallan fg = ri->ri_devcmap[(attr >> 24) & 0x0f]; 13617bbb04c1Smacallan 13627bbb04c1Smacallan summitfb_write_mode(sc, 0x050000c0); 1363d2ce6272Smacallan summitfb_wait_fifo(sc, 8); 13641558ebccSmacallan summitfb_write4(sc, VISFX_IBO, RopSrc); 13657bbb04c1Smacallan summitfb_write4(sc, VISFX_FG_COLOUR, fg); 13667bbb04c1Smacallan summitfb_write4(sc, VISFX_BG_COLOUR, bg); 13677bbb04c1Smacallan 13687bbb04c1Smacallan i = c - font->firstchar; 13697bbb04c1Smacallan xs = sc->sc_font_start + (i % sc->sc_cols) * sc->sc_font->fontwidth; 13707bbb04c1Smacallan ys = (i / sc->sc_cols) * sc->sc_font->fontheight; 13717bbb04c1Smacallan 1372d2ce6272Smacallan summitfb_wait_fifo(sc, 8); 13737bbb04c1Smacallan summitfb_write4(sc, VISFX_COPY_SRC, (xs << 16) | ys); 13747bbb04c1Smacallan summitfb_write4(sc, VISFX_COPY_WH, (wi << 16) | he); 13757bbb04c1Smacallan summitfb_write4(sc, VISFX_COPY_DST, (x << 16) | y); 13767bbb04c1Smacallan 13777bbb04c1Smacallan } 13787bbb04c1Smacallan 13797bbb04c1Smacallan static void 13809ad1548eSmacallan summitfb_putchar_aa(void *cookie, int row, int col, u_int c, long attr) 13819ad1548eSmacallan { 13829ad1548eSmacallan struct rasops_info *ri = cookie; 13839ad1548eSmacallan struct wsdisplay_font *font = PICK_FONT(ri, c); 13849ad1548eSmacallan struct vcons_screen *scr = ri->ri_hw; 13859ad1548eSmacallan struct summitfb_softc *sc = scr->scr_cookie; 1386*86973f0aSmacallan int x, y, wi, he, rv = GC_NOPE, i, j; 1387*86973f0aSmacallan uint32_t bg, fg, tmp; 1388*86973f0aSmacallan uint8_t *data; 13899ad1548eSmacallan 13909ad1548eSmacallan if (sc->sc_mode != WSDISPLAYIO_MODE_EMUL) 13919ad1548eSmacallan return; 13929ad1548eSmacallan 13939ad1548eSmacallan if (!CHAR_IN_FONT(c, font)) 13949ad1548eSmacallan return; 13959ad1548eSmacallan 13969ad1548eSmacallan if (row == ri->ri_crow && col == ri->ri_ccol) { 13979ad1548eSmacallan ri->ri_flg &= ~RI_CURSOR; 13989ad1548eSmacallan } 13999ad1548eSmacallan 14009ad1548eSmacallan wi = font->fontwidth; 14019ad1548eSmacallan he = font->fontheight; 14029ad1548eSmacallan 14039ad1548eSmacallan x = ri->ri_xorigin + col * wi; 14049ad1548eSmacallan y = ri->ri_yorigin + row * he; 14059ad1548eSmacallan 14069ad1548eSmacallan bg = ri->ri_devcmap[(attr >> 16) & 0xf]; 14079ad1548eSmacallan 14089ad1548eSmacallan if (c == 0x20) { 14099ad1548eSmacallan summitfb_rectfill(sc, x, y, wi, he, bg); 14109ad1548eSmacallan return; 14119ad1548eSmacallan } 14129ad1548eSmacallan 14139ad1548eSmacallan rv = glyphcache_try(&sc->sc_gc, c, x, y, attr); 14149ad1548eSmacallan if (rv == GC_OK) 14159ad1548eSmacallan return; 14169ad1548eSmacallan 1417*86973f0aSmacallan /* 1418*86973f0aSmacallan * first we clear the background - we should be able to use the CBR 1419*86973f0aSmacallan * register as constant background but so far I couldn't make that work 1420*86973f0aSmacallan */ 1421*86973f0aSmacallan summitfb_rectfill(sc, x, y, wi, he, bg); 1422*86973f0aSmacallan 1423*86973f0aSmacallan /* 1424*86973f0aSmacallan * if we ever figure out how to use a constant background colour we can 1425*86973f0aSmacallan * skip the read mode setting 1426*86973f0aSmacallan */ 1427*86973f0aSmacallan summitfb_read_mode(sc, OTC01 | BIN332F | BUFovl); 1428*86973f0aSmacallan summitfb_write_mode(sc, OTC01 | BIN332F | BUFovl); 1429*86973f0aSmacallan /* we need the foreground colour as full RGB8 */ 1430*86973f0aSmacallan fg = sc->sc_palette[(attr >> 24) & 0xf]; 1431*86973f0aSmacallan 1432*86973f0aSmacallan /* 1433*86973f0aSmacallan * set the blending equation to 1434*86973f0aSmacallan * src_color * src_alpha + dst_color * (1 - src_alpha) 1435*86973f0aSmacallan */ 1436*86973f0aSmacallan summitfb_write4(sc, VISFX_IBO, 1437*86973f0aSmacallan IBO_ADD | SRC(IBO_SRC) | DST(IBO_ONE_MINUS_SRC)); 1438*86973f0aSmacallan 1439*86973f0aSmacallan /* get the glyph */ 1440*86973f0aSmacallan data = WSFONT_GLYPH(c, font); 1441*86973f0aSmacallan for (i = 0; i < he; i++) { 1442*86973f0aSmacallan /* 1443*86973f0aSmacallan * make some room in the pipeline 1444*86973f0aSmacallan * with just plain ROPs we can just hammer the FIFO without 1445*86973f0aSmacallan * having to worry about overflowing it but I suspect with 1446*86973f0aSmacallan * alpha blending enabled things may be a little slower 1447*86973f0aSmacallan */ 1448*86973f0aSmacallan summitfb_wait_fifo(sc, wi * 2); 1449*86973f0aSmacallan /* start a new line */ 1450*86973f0aSmacallan summitfb_write4(sc, VISFX_VRAM_WRITE_DEST, ((y + i) << 16) | x); 1451*86973f0aSmacallan for (j = 0; j < wi; j++) { 1452*86973f0aSmacallan tmp = *data; 1453*86973f0aSmacallan /* alpha & RGB -> ARGB */ 1454*86973f0aSmacallan summitfb_write4(sc, VISFX_VRAM_WRITE_DATA_INCRX, 1455*86973f0aSmacallan (tmp << 24) | fg); 1456*86973f0aSmacallan data++; 1457*86973f0aSmacallan } 1458*86973f0aSmacallan } 14599ad1548eSmacallan 14609ad1548eSmacallan if (rv == GC_ADD) 14619ad1548eSmacallan glyphcache_add(&sc->sc_gc, c, x, y); 14629ad1548eSmacallan } 14639ad1548eSmacallan 14649ad1548eSmacallan static void 14659ad1548eSmacallan summitfb_copycols(void *cookie, int row, int srccol, int dstcol, int ncols) 14669ad1548eSmacallan { 14679ad1548eSmacallan struct rasops_info *ri = cookie; 14689ad1548eSmacallan struct vcons_screen *scr = ri->ri_hw; 14699ad1548eSmacallan struct summitfb_softc *sc = scr->scr_cookie; 14709ad1548eSmacallan int32_t xs, xd, y, width, height; 14719ad1548eSmacallan 147238fdde35Sskrll if (sc->sc_locked == 0 && sc->sc_mode == WSDISPLAYIO_MODE_EMUL) { 14731558ebccSmacallan 14749ad1548eSmacallan if (ri->ri_crow == row && 1475c84cdbddSriastradh ri->ri_ccol >= srccol && ri->ri_ccol < (srccol + ncols) && 14769ad1548eSmacallan (ri->ri_flg & RI_CURSOR)) { 14779ad1548eSmacallan summitfb_nuke_cursor(ri); 14789ad1548eSmacallan } 14791558ebccSmacallan 14809ad1548eSmacallan xs = ri->ri_xorigin + ri->ri_font->fontwidth * srccol; 14819ad1548eSmacallan xd = ri->ri_xorigin + ri->ri_font->fontwidth * dstcol; 14829ad1548eSmacallan y = ri->ri_yorigin + ri->ri_font->fontheight * row; 14839ad1548eSmacallan width = ri->ri_font->fontwidth * ncols; 14849ad1548eSmacallan height = ri->ri_font->fontheight; 14859ad1548eSmacallan summitfb_bitblt(sc, xs, y, xd, y, width, height, RopSrc); 14861dd4359cSmacallan 14879ad1548eSmacallan if (ri->ri_crow == row && 1488c84cdbddSriastradh ri->ri_ccol >= dstcol && ri->ri_ccol < (dstcol + ncols)) 14899ad1548eSmacallan ri->ri_flg &= ~RI_CURSOR; 14909ad1548eSmacallan } 14919ad1548eSmacallan } 14929ad1548eSmacallan 14939ad1548eSmacallan static void 1494c84cdbddSriastradh summitfb_erasecols(void *cookie, int row, int startcol, int ncols, 1495c84cdbddSriastradh long fillattr) 14969ad1548eSmacallan { 14979ad1548eSmacallan struct rasops_info *ri = cookie; 14989ad1548eSmacallan struct vcons_screen *scr = ri->ri_hw; 14999ad1548eSmacallan struct summitfb_softc *sc = scr->scr_cookie; 15009ad1548eSmacallan int32_t x, y, width, height, fg, bg, ul; 15019ad1548eSmacallan 150238fdde35Sskrll if (sc->sc_locked == 0 && sc->sc_mode == WSDISPLAYIO_MODE_EMUL) { 15039ad1548eSmacallan x = ri->ri_xorigin + ri->ri_font->fontwidth * startcol; 15049ad1548eSmacallan y = ri->ri_yorigin + ri->ri_font->fontheight * row; 15059ad1548eSmacallan width = ri->ri_font->fontwidth * ncols; 15069ad1548eSmacallan height = ri->ri_font->fontheight; 15079ad1548eSmacallan rasops_unpack_attr(fillattr, &fg, &bg, &ul); 15089ad1548eSmacallan 15099ad1548eSmacallan summitfb_rectfill(sc, x, y, width, height, ri->ri_devcmap[bg]); 15101dd4359cSmacallan 15119ad1548eSmacallan if (ri->ri_crow == row && 1512c84cdbddSriastradh ri->ri_ccol >= startcol && 1513c84cdbddSriastradh ri->ri_ccol < (startcol + ncols)) 15149ad1548eSmacallan ri->ri_flg &= ~RI_CURSOR; 15159ad1548eSmacallan } 15169ad1548eSmacallan } 15179ad1548eSmacallan 15189ad1548eSmacallan static void 15199ad1548eSmacallan summitfb_copyrows(void *cookie, int srcrow, int dstrow, int nrows) 15209ad1548eSmacallan { 15219ad1548eSmacallan struct rasops_info *ri = cookie; 15229ad1548eSmacallan struct vcons_screen *scr = ri->ri_hw; 15239ad1548eSmacallan struct summitfb_softc *sc = scr->scr_cookie; 15249ad1548eSmacallan int32_t x, ys, yd, width, height; 15259ad1548eSmacallan 152638fdde35Sskrll if (sc->sc_locked == 0 && sc->sc_mode == WSDISPLAYIO_MODE_EMUL) { 15271558ebccSmacallan 1528c84cdbddSriastradh if (ri->ri_crow >= srcrow && ri->ri_crow < (srcrow + nrows) && 15299ad1548eSmacallan (ri->ri_flg & RI_CURSOR)) { 15309ad1548eSmacallan summitfb_nuke_cursor(ri); 15319ad1548eSmacallan } 15321558ebccSmacallan 15339ad1548eSmacallan x = ri->ri_xorigin; 15349ad1548eSmacallan ys = ri->ri_yorigin + ri->ri_font->fontheight * srcrow; 15359ad1548eSmacallan yd = ri->ri_yorigin + ri->ri_font->fontheight * dstrow; 15369ad1548eSmacallan width = ri->ri_emuwidth; 15379ad1548eSmacallan height = ri->ri_font->fontheight * nrows; 15389ad1548eSmacallan summitfb_bitblt(sc, x, ys, x, yd, width, height, RopSrc); 15391dd4359cSmacallan 15409ad1548eSmacallan if (ri->ri_crow >= dstrow && ri->ri_crow < (dstrow + nrows)) 15419ad1548eSmacallan ri->ri_flg &= ~RI_CURSOR; 15429ad1548eSmacallan } 15439ad1548eSmacallan } 15449ad1548eSmacallan 15459ad1548eSmacallan static void 15469ad1548eSmacallan summitfb_eraserows(void *cookie, int row, int nrows, long fillattr) 15479ad1548eSmacallan { 15489ad1548eSmacallan struct rasops_info *ri = cookie; 15499ad1548eSmacallan struct vcons_screen *scr = ri->ri_hw; 15509ad1548eSmacallan struct summitfb_softc *sc = scr->scr_cookie; 15519ad1548eSmacallan int32_t x, y, width, height, fg, bg, ul; 15529ad1548eSmacallan 155338fdde35Sskrll if (sc->sc_locked == 0 && sc->sc_mode == WSDISPLAYIO_MODE_EMUL) { 15549ad1548eSmacallan x = ri->ri_xorigin; 15559ad1548eSmacallan y = ri->ri_yorigin + ri->ri_font->fontheight * row; 15569ad1548eSmacallan width = ri->ri_emuwidth; 15579ad1548eSmacallan height = ri->ri_font->fontheight * nrows; 15589ad1548eSmacallan rasops_unpack_attr(fillattr, &fg, &bg, &ul); 15599ad1548eSmacallan 15609ad1548eSmacallan summitfb_rectfill(sc, x, y, width, height, ri->ri_devcmap[bg]); 15619ad1548eSmacallan 15629ad1548eSmacallan if (ri->ri_crow >= row && ri->ri_crow < (row + nrows)) 15639ad1548eSmacallan ri->ri_flg &= ~RI_CURSOR; 15649ad1548eSmacallan } 15659ad1548eSmacallan } 15669ad1548eSmacallan 15679ad1548eSmacallan static void 15689ad1548eSmacallan summitfb_move_cursor(struct summitfb_softc *sc, int x, int y) 15699ad1548eSmacallan { 15709ad1548eSmacallan uint32_t pos; 15719ad1548eSmacallan 15729ad1548eSmacallan sc->sc_cursor_x = x; 15739ad1548eSmacallan x -= sc->sc_hot_x; 15749ad1548eSmacallan sc->sc_cursor_y = y; 15759ad1548eSmacallan y -= sc->sc_hot_y; 15769ad1548eSmacallan 157738fdde35Sskrll if (x < 0) 157838fdde35Sskrll x = 0x1000 - x; 157938fdde35Sskrll if (y < 0) 158038fdde35Sskrll y = 0x1000 - y; 15819ad1548eSmacallan pos = (x << 16) | y; 158238fdde35Sskrll if (sc->sc_enabled) 158338fdde35Sskrll pos |= 0x80000000; 1584f7327fc7Smacallan summitfb_write4(sc, VISFX_CURSOR_POS, pos); 15859ad1548eSmacallan } 15869ad1548eSmacallan 15879ad1548eSmacallan static int 15889ad1548eSmacallan summitfb_do_cursor(struct summitfb_softc *sc, struct wsdisplay_cursor *cur) 15899ad1548eSmacallan { 15909ad1548eSmacallan 1591c84cdbddSriastradh if (cur->which & WSDISPLAY_CURSOR_DOCUR) { 15929ad1548eSmacallan sc->sc_enabled = cur->enable; 15939ad1548eSmacallan cur->which |= WSDISPLAY_CURSOR_DOPOS; 15949ad1548eSmacallan } 15959ad1548eSmacallan if (cur->which & WSDISPLAY_CURSOR_DOHOT) { 15969ad1548eSmacallan sc->sc_hot_x = cur->hot.x; 15979ad1548eSmacallan sc->sc_hot_y = cur->hot.y; 15989ad1548eSmacallan cur->which |= WSDISPLAY_CURSOR_DOPOS; 15999ad1548eSmacallan } 16009ad1548eSmacallan if (cur->which & WSDISPLAY_CURSOR_DOPOS) { 16019ad1548eSmacallan summitfb_move_cursor(sc, cur->pos.x, cur->pos.y); 16029ad1548eSmacallan } 16039ad1548eSmacallan if (cur->which & WSDISPLAY_CURSOR_DOCMAP) { 1604f7327fc7Smacallan uint32_t rgb; 16059ad1548eSmacallan uint8_t r[2], g[2], b[2]; 16069ad1548eSmacallan 16079ad1548eSmacallan copyin(cur->cmap.blue, b, 2); 16089ad1548eSmacallan copyin(cur->cmap.green, g, 2); 16099ad1548eSmacallan copyin(cur->cmap.red, r, 2); 1610f7327fc7Smacallan summitfb_write4(sc, VISFX_CURSOR_INDEX, 0); 1611f7327fc7Smacallan rgb = r[0] << 16 | g[0] << 8 | b[0]; 16123f08161dSmacallan summitfb_write4(sc, VISFX_CURSOR_BG, rgb); 1613f7327fc7Smacallan rgb = r[1] << 16 | g[1] << 8 | b[1]; 16143f08161dSmacallan summitfb_write4(sc, VISFX_CURSOR_FG, rgb); 16151dd4359cSmacallan 16169ad1548eSmacallan } 16179ad1548eSmacallan if (cur->which & WSDISPLAY_CURSOR_DOSHAPE) { 1618f7327fc7Smacallan 16199ad1548eSmacallan uint32_t buffer[128], latch, tmp; 16209ad1548eSmacallan int i; 16219ad1548eSmacallan 16229ad1548eSmacallan copyin(cur->mask, buffer, 512); 1623f7327fc7Smacallan summitfb_write4(sc, VISFX_CURSOR_INDEX, 0); 16249ad1548eSmacallan for (i = 0; i < 128; i += 2) { 16259ad1548eSmacallan latch = 0; 16269ad1548eSmacallan tmp = buffer[i] & 0x80808080; 16279ad1548eSmacallan latch |= tmp >> 7; 16289ad1548eSmacallan tmp = buffer[i] & 0x40404040; 16299ad1548eSmacallan latch |= tmp >> 5; 16309ad1548eSmacallan tmp = buffer[i] & 0x20202020; 16319ad1548eSmacallan latch |= tmp >> 3; 16329ad1548eSmacallan tmp = buffer[i] & 0x10101010; 16339ad1548eSmacallan latch |= tmp >> 1; 16349ad1548eSmacallan tmp = buffer[i] & 0x08080808; 16359ad1548eSmacallan latch |= tmp << 1; 16369ad1548eSmacallan tmp = buffer[i] & 0x04040404; 16379ad1548eSmacallan latch |= tmp << 3; 16389ad1548eSmacallan tmp = buffer[i] & 0x02020202; 16399ad1548eSmacallan latch |= tmp << 5; 16409ad1548eSmacallan tmp = buffer[i] & 0x01010101; 16419ad1548eSmacallan latch |= tmp << 7; 1642f7327fc7Smacallan summitfb_write4(sc, VISFX_CURSOR_DATA, latch); 16439ad1548eSmacallan latch = 0; 16449ad1548eSmacallan tmp = buffer[i + 1] & 0x80808080; 16459ad1548eSmacallan latch |= tmp >> 7; 16469ad1548eSmacallan tmp = buffer[i + 1] & 0x40404040; 16479ad1548eSmacallan latch |= tmp >> 5; 16489ad1548eSmacallan tmp = buffer[i + 1] & 0x20202020; 16499ad1548eSmacallan latch |= tmp >> 3; 16509ad1548eSmacallan tmp = buffer[i + 1] & 0x10101010; 16519ad1548eSmacallan latch |= tmp >> 1; 16529ad1548eSmacallan tmp = buffer[i + 1] & 0x08080808; 16539ad1548eSmacallan latch |= tmp << 1; 16549ad1548eSmacallan tmp = buffer[i + 1] & 0x04040404; 16559ad1548eSmacallan latch |= tmp << 3; 16569ad1548eSmacallan tmp = buffer[i + 1] & 0x02020202; 16579ad1548eSmacallan latch |= tmp << 5; 16589ad1548eSmacallan tmp = buffer[i + 1] & 0x01010101; 16599ad1548eSmacallan latch |= tmp << 7; 1660f7327fc7Smacallan summitfb_write4(sc, VISFX_CURSOR_DATA, latch); 16619ad1548eSmacallan } 16629ad1548eSmacallan 1663f7327fc7Smacallan summitfb_write4(sc, VISFX_CURSOR_INDEX, 0x80); 16649ad1548eSmacallan copyin(cur->image, buffer, 512); 16659ad1548eSmacallan for (i = 0; i < 128; i += 2) { 16669ad1548eSmacallan latch = 0; 16679ad1548eSmacallan tmp = buffer[i] & 0x80808080; 16689ad1548eSmacallan latch |= tmp >> 7; 16699ad1548eSmacallan tmp = buffer[i] & 0x40404040; 16709ad1548eSmacallan latch |= tmp >> 5; 16719ad1548eSmacallan tmp = buffer[i] & 0x20202020; 16729ad1548eSmacallan latch |= tmp >> 3; 16739ad1548eSmacallan tmp = buffer[i] & 0x10101010; 16749ad1548eSmacallan latch |= tmp >> 1; 16759ad1548eSmacallan tmp = buffer[i] & 0x08080808; 16769ad1548eSmacallan latch |= tmp << 1; 16779ad1548eSmacallan tmp = buffer[i] & 0x04040404; 16789ad1548eSmacallan latch |= tmp << 3; 16799ad1548eSmacallan tmp = buffer[i] & 0x02020202; 16809ad1548eSmacallan latch |= tmp << 5; 16819ad1548eSmacallan tmp = buffer[i] & 0x01010101; 16829ad1548eSmacallan latch |= tmp << 7; 1683f7327fc7Smacallan summitfb_write4(sc, VISFX_CURSOR_DATA, latch); 16849ad1548eSmacallan latch = 0; 16859ad1548eSmacallan tmp = buffer[i + 1] & 0x80808080; 16869ad1548eSmacallan latch |= tmp >> 7; 16879ad1548eSmacallan tmp = buffer[i + 1] & 0x40404040; 16889ad1548eSmacallan latch |= tmp >> 5; 16899ad1548eSmacallan tmp = buffer[i + 1] & 0x20202020; 16909ad1548eSmacallan latch |= tmp >> 3; 16919ad1548eSmacallan tmp = buffer[i + 1] & 0x10101010; 16929ad1548eSmacallan latch |= tmp >> 1; 16939ad1548eSmacallan tmp = buffer[i + 1] & 0x08080808; 16949ad1548eSmacallan latch |= tmp << 1; 16959ad1548eSmacallan tmp = buffer[i + 1] & 0x04040404; 16969ad1548eSmacallan latch |= tmp << 3; 16979ad1548eSmacallan tmp = buffer[i + 1] & 0x02020202; 16989ad1548eSmacallan latch |= tmp << 5; 16999ad1548eSmacallan tmp = buffer[i + 1] & 0x01010101; 17009ad1548eSmacallan latch |= tmp << 7; 1701f7327fc7Smacallan summitfb_write4(sc, VISFX_CURSOR_DATA, latch); 17029ad1548eSmacallan } 17039ad1548eSmacallan } 17049ad1548eSmacallan 17059ad1548eSmacallan return 0; 17069ad1548eSmacallan } 17079ad1548eSmacallan 17089ad1548eSmacallan static void 17099ad1548eSmacallan summitfb_set_video(struct summitfb_softc *sc, int on) 17109ad1548eSmacallan { 1711c84cdbddSriastradh 17129ad1548eSmacallan if (sc->sc_video_on == on) 17139ad1548eSmacallan return; 17149ad1548eSmacallan 17159ad1548eSmacallan sc->sc_video_on = on; 17169ad1548eSmacallan 17179ad1548eSmacallan summitfb_wait(sc); 17189ad1548eSmacallan if (on) { 1719bc72ec74Smacallan summitfb_write4(sc, VISFX_MPC, MPC_VIDEO_ON); 17209ad1548eSmacallan } else { 1721bc72ec74Smacallan summitfb_write4(sc, VISFX_MPC, MPC_VSYNC_OFF | MPC_HSYNC_OFF); 17229ad1548eSmacallan } 17239ad1548eSmacallan } 17246fcfb3adSmacallan 17256fcfb3adSmacallan static void 17266fcfb3adSmacallan summitfb_copyfont(struct summitfb_softc *sc) 17276fcfb3adSmacallan { 17286fcfb3adSmacallan struct sti_font *fp = &sc->sc_scr.scr_curfont; 17296fcfb3adSmacallan uint8_t *font = sc->sc_scr.scr_romfont; 17306fcfb3adSmacallan uint8_t *fontbuf, *fontdata, *src, *dst; 17316fcfb3adSmacallan struct wsdisplay_font *f; 17326fcfb3adSmacallan int bufsize, i, si; 17336fcfb3adSmacallan 173438fdde35Sskrll if (font == NULL) 173538fdde35Sskrll return; 173697eb7ca2Sskrll 17376fcfb3adSmacallan bufsize = sizeof(struct wsdisplay_font) + 32 + fp->bpc * (fp->last - fp->first); 1738b04ca5b4Smacallan DPRINTF(("%s: %dx%d %d\n", __func__, fp->width, fp->height, bufsize)); 173997eb7ca2Sskrll fontbuf = kmem_alloc(bufsize, KM_SLEEP); 17406fcfb3adSmacallan f = (struct wsdisplay_font *)fontbuf; 17416fcfb3adSmacallan f->name = fontbuf + sizeof(struct wsdisplay_font); 17426fcfb3adSmacallan fontdata = fontbuf + sizeof(struct wsdisplay_font) + 32; 17436fcfb3adSmacallan strcpy(fontbuf + sizeof(struct wsdisplay_font), "HP ROM"); 17446fcfb3adSmacallan f->firstchar = fp->first; 17456fcfb3adSmacallan f->numchars = (fp->last + 1) - fp->first; 17466fcfb3adSmacallan f->encoding = WSDISPLAY_FONTENC_ISO; 17476fcfb3adSmacallan f->fontwidth = fp->width; 17486fcfb3adSmacallan f->fontheight = fp->height; 17496fcfb3adSmacallan f->stride = (fp->width + 7) >> 3; 17506fcfb3adSmacallan f->bitorder = WSDISPLAY_FONTORDER_L2R; 17516fcfb3adSmacallan f->byteorder = WSDISPLAY_FONTORDER_L2R; 17526fcfb3adSmacallan f->data = fontdata; 17536fcfb3adSmacallan /* skip over font struct */ 17546fcfb3adSmacallan font += sizeof(struct sti_font); 17556fcfb3adSmacallan /* now copy and rearrange the glyphs into ISO order */ 17566fcfb3adSmacallan /* first, copy the characters up to 0x7f */ 17576fcfb3adSmacallan memcpy(fontdata, font, (0x80 - fp->first) * fp->bpc); 17586fcfb3adSmacallan /* zero 0x80 to 0x9f */ 17596fcfb3adSmacallan memset(fontdata + 0x80 * fp->bpc, 0, 0x20 * fp->bpc); 17606fcfb3adSmacallan /* rearrange 0xa0 till last */ 17616fcfb3adSmacallan for (i = 0xa0; i < (fp->last + 1); i++) { 17626fcfb3adSmacallan dst = fontdata + fp->bpc * i; 17636fcfb3adSmacallan si = sti_unitoroman[i - 0xa0]; 17646fcfb3adSmacallan if (si != 0) { 17656fcfb3adSmacallan src = font + fp->bpc * si; 17666fcfb3adSmacallan memcpy(dst, src, fp->bpc); 17676fcfb3adSmacallan } else { 17686fcfb3adSmacallan /* no mapping - zeeo this cell */ 17696fcfb3adSmacallan memset(dst, 0, fp->bpc); 17706fcfb3adSmacallan } 17716fcfb3adSmacallan } 17726fcfb3adSmacallan wsfont_add(f, 0); 17736fcfb3adSmacallan } 1774