xref: /openbsd-src/sys/arch/loongson/dev/sisfb.c (revision 63294167c4eaf171adac4fb2d8cc6d072a13af61)
1*63294167Skettenis /*	$OpenBSD: sisfb.c,v 1.10 2022/07/15 17:57:26 kettenis Exp $	*/
295b699e3Smiod 
395b699e3Smiod /*
495b699e3Smiod  * Copyright (c) 2010 Miodrag Vallat.
595b699e3Smiod  *
695b699e3Smiod  * Permission to use, copy, modify, and distribute this software for any
795b699e3Smiod  * purpose with or without fee is hereby granted, provided that the above
895b699e3Smiod  * copyright notice and this permission notice appear in all copies.
995b699e3Smiod  *
1095b699e3Smiod  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
1195b699e3Smiod  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
1295b699e3Smiod  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
1395b699e3Smiod  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
1495b699e3Smiod  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
1595b699e3Smiod  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
1695b699e3Smiod  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
1795b699e3Smiod  */
1895b699e3Smiod 
1995b699e3Smiod /*
2095b699e3Smiod  * Minimalistic driver for the SIS315 Pro frame buffer found on the
2195b699e3Smiod  * Lemote Fuloong 2F systems.
2236fd90dcSjsg  * Does not support acceleration, mode change, secondary output, or
2395b699e3Smiod  * anything fancy.
2495b699e3Smiod  */
2595b699e3Smiod 
2695b699e3Smiod #include <sys/param.h>
2795b699e3Smiod #include <sys/systm.h>
2895b699e3Smiod #include <sys/device.h>
2995b699e3Smiod 
3095b699e3Smiod #include <machine/bus.h>
3195b699e3Smiod #include <machine/cpu.h>
3295b699e3Smiod 
3395b699e3Smiod #include <uvm/uvm_extern.h>
3495b699e3Smiod 
3595b699e3Smiod #include <dev/pci/pcireg.h>
3695b699e3Smiod #include <dev/pci/pcivar.h>
3795b699e3Smiod #include <dev/pci/pcidevs.h>
3895b699e3Smiod 
3995b699e3Smiod #include <dev/ic/mc6845.h>
4095b699e3Smiod 
4195b699e3Smiod #include <dev/wscons/wsconsio.h>
4295b699e3Smiod #include <dev/wscons/wsdisplayvar.h>
4395b699e3Smiod #include <dev/rasops/rasops.h>
4495b699e3Smiod 
4595b699e3Smiod struct sisfb_softc;
4695b699e3Smiod 
4795b699e3Smiod /* minimal frame buffer information, suitable for early console */
4895b699e3Smiod struct sisfb {
4995b699e3Smiod 	struct sisfb_softc	*sc;
5095b699e3Smiod 	struct rasops_info	 ri;
5195b699e3Smiod 	uint8_t			 cmap[256 * 3];
5295b699e3Smiod 
5395b699e3Smiod 	bus_space_tag_t		 fbt;
5495b699e3Smiod 	bus_space_handle_t	 fbh;
5595b699e3Smiod 
5695b699e3Smiod 	bus_space_tag_t		 mmiot;
5795b699e3Smiod 	bus_space_handle_t	 mmioh;
5895b699e3Smiod 
5995b699e3Smiod 	bus_space_tag_t		 iot;
6095b699e3Smiod 	bus_space_handle_t	 ioh;
6195b699e3Smiod 
6295b699e3Smiod 	struct wsscreen_descr	 wsd;
6395b699e3Smiod };
6495b699e3Smiod 
6595b699e3Smiod struct sisfb_softc {
6695b699e3Smiod 	struct device		 sc_dev;
6795b699e3Smiod 	struct sisfb		*sc_fb;
6895b699e3Smiod 	struct sisfb		 sc_fb_store;
6995b699e3Smiod 
7095b699e3Smiod 	struct wsscreen_list	 sc_wsl;
7195b699e3Smiod 	struct wsscreen_descr	*sc_scrlist[1];
7295b699e3Smiod 	int			 sc_nscr;
7395b699e3Smiod };
7495b699e3Smiod 
7595b699e3Smiod int	sisfb_match(struct device *, void *, void *);
7695b699e3Smiod void	sisfb_attach(struct device *, struct device *, void *);
7795b699e3Smiod 
7895b699e3Smiod const struct cfattach sisfb_ca = {
7995b699e3Smiod 	sizeof(struct sisfb_softc), sisfb_match, sisfb_attach
8095b699e3Smiod };
8195b699e3Smiod 
8295b699e3Smiod struct cfdriver sisfb_cd = {
8395b699e3Smiod 	NULL, "sisfb", DV_DULL
8495b699e3Smiod };
8595b699e3Smiod 
8695b699e3Smiod int	sisfb_alloc_screen(void *, const struct wsscreen_descr *, void **, int *,
87e0c3e559Sjsg 	    int *, uint32_t *);
8895b699e3Smiod void	sisfb_free_screen(void *, void *);
8995b699e3Smiod int	sisfb_ioctl(void *, u_long, caddr_t, int, struct proc *);
9083275742Smiod int	sisfb_list_font(void *, struct wsdisplay_font *);
9183275742Smiod int	sisfb_load_font(void *, void *, struct wsdisplay_font *);
9283275742Smiod paddr_t	sisfb_mmap(void *, off_t, int);
9395b699e3Smiod int	sisfb_show_screen(void *, void *, int, void (*)(void *, int, int),
9495b699e3Smiod 	    void *);
9595b699e3Smiod 
9695b699e3Smiod struct wsdisplay_accessops sisfb_accessops = {
9787eec248Smiod 	.ioctl = sisfb_ioctl,
9887eec248Smiod 	.mmap = sisfb_mmap,
9987eec248Smiod 	.alloc_screen = sisfb_alloc_screen,
10087eec248Smiod 	.free_screen = sisfb_free_screen,
10183275742Smiod 	.show_screen = sisfb_show_screen,
10283275742Smiod 	.load_font = sisfb_load_font,
10383275742Smiod 	.list_font = sisfb_list_font
10495b699e3Smiod };
10595b699e3Smiod 
10695b699e3Smiod int	sisfb_getcmap(uint8_t *, struct wsdisplay_cmap *);
10795b699e3Smiod void	sisfb_loadcmap(struct sisfb *, int, int);
10895b699e3Smiod int	sisfb_putcmap(uint8_t *, struct wsdisplay_cmap *);
10995b699e3Smiod int	sisfb_setup(struct sisfb *);
11095b699e3Smiod 
11195b699e3Smiod static struct sisfb sisfbcn;
11295b699e3Smiod 
11395b699e3Smiod const struct pci_matchid sisfb_devices[] = {
11495b699e3Smiod 	{ PCI_VENDOR_SIS, PCI_PRODUCT_SIS_315PRO_VGA }
11595b699e3Smiod };
11695b699e3Smiod 
11795b699e3Smiod /*
11895b699e3Smiod  * Control Register access
11995b699e3Smiod  *
12095b699e3Smiod  * These are 8 bit registers; the choice of larger width types is intentional.
12195b699e3Smiod  */
12295b699e3Smiod 
12395b699e3Smiod #define	SIS_VGA_PORT_OFFSET	0x380
12495b699e3Smiod 
12595b699e3Smiod #define	SEQ_ADDR		(0x3c4 - SIS_VGA_PORT_OFFSET)
12695b699e3Smiod #define	SEQ_DATA		(0x3c5 - SIS_VGA_PORT_OFFSET)
12795b699e3Smiod #define	DAC_ADDR		(0x3c8 - SIS_VGA_PORT_OFFSET)
12895b699e3Smiod #define	DAC_DATA		(0x3c9 - SIS_VGA_PORT_OFFSET)
12995b699e3Smiod #undef	CRTC_ADDR
13095b699e3Smiod #define	CRTC_ADDR		(0x3d4 - SIS_VGA_PORT_OFFSET)
13195b699e3Smiod #define	CRTC_DATA		(0x3d5 - SIS_VGA_PORT_OFFSET)
13295b699e3Smiod 
13395b699e3Smiod static inline uint sisfb_crtc_read(struct sisfb *, uint);
13495b699e3Smiod static inline uint sisfb_seq_read(struct sisfb *, uint);
13595b699e3Smiod static inline void sisfb_seq_write(struct sisfb *, uint, uint);
13695b699e3Smiod 
13795b699e3Smiod static inline uint
sisfb_crtc_read(struct sisfb * fb,uint idx)13895b699e3Smiod sisfb_crtc_read(struct sisfb *fb, uint idx)
13995b699e3Smiod {
14095b699e3Smiod 	uint val;
14195b699e3Smiod 	bus_space_write_1(fb->iot, fb->ioh, CRTC_ADDR, idx);
14295b699e3Smiod 	val = bus_space_read_1(fb->iot, fb->ioh, CRTC_DATA);
14395b699e3Smiod #ifdef SIS_DEBUG
14495b699e3Smiod 	printf("CRTC %04x -> %02x\n", idx, val);
14595b699e3Smiod #endif
14695b699e3Smiod 	return val;
14795b699e3Smiod }
14895b699e3Smiod 
14995b699e3Smiod static inline uint
sisfb_seq_read(struct sisfb * fb,uint idx)15095b699e3Smiod sisfb_seq_read(struct sisfb *fb, uint idx)
15195b699e3Smiod {
15295b699e3Smiod 	uint val;
15395b699e3Smiod 	bus_space_write_1(fb->iot, fb->ioh, SEQ_ADDR, idx);
15495b699e3Smiod 	val = bus_space_read_1(fb->iot, fb->ioh, SEQ_DATA);
15595b699e3Smiod #ifdef SIS_DEBUG
15695b699e3Smiod 	printf("SEQ %04x -> %02x\n", idx, val);
15795b699e3Smiod #endif
15895b699e3Smiod 	return val;
15995b699e3Smiod }
16095b699e3Smiod 
16195b699e3Smiod static inline void
sisfb_seq_write(struct sisfb * fb,uint idx,uint val)16295b699e3Smiod sisfb_seq_write(struct sisfb *fb, uint idx, uint val)
16395b699e3Smiod {
16495b699e3Smiod #ifdef SIS_DEBUG
16595b699e3Smiod 	printf("SEQ %04x <- %02x\n", idx, val);
16695b699e3Smiod #endif
16795b699e3Smiod 	bus_space_write_1(fb->iot, fb->ioh, SEQ_ADDR, idx);
16895b699e3Smiod 	bus_space_write_1(fb->iot, fb->ioh, SEQ_DATA, val);
16995b699e3Smiod }
17095b699e3Smiod 
17195b699e3Smiod int
sisfb_match(struct device * parent,void * vcf,void * aux)17295b699e3Smiod sisfb_match(struct device *parent, void *vcf, void *aux)
17395b699e3Smiod {
17495b699e3Smiod 	struct pci_attach_args *pa = (struct pci_attach_args *)aux;
17595b699e3Smiod 
17695b699e3Smiod 	return pci_matchbyid(pa, sisfb_devices, nitems(sisfb_devices));
17795b699e3Smiod }
17895b699e3Smiod 
17995b699e3Smiod void
sisfb_attach(struct device * parent,struct device * self,void * aux)18095b699e3Smiod sisfb_attach(struct device *parent, struct device *self, void *aux)
18195b699e3Smiod {
18295b699e3Smiod 	struct sisfb_softc *sc = (struct sisfb_softc *)self;
18395b699e3Smiod 	struct pci_attach_args *pa = (struct pci_attach_args *)aux;
18495b699e3Smiod 	struct wsemuldisplaydev_attach_args waa;
18595b699e3Smiod 	bus_space_tag_t fbt, mmiot, iot;
18695b699e3Smiod 	bus_space_handle_t fbh, mmioh, ioh;
18795b699e3Smiod 	bus_size_t fbsize, mmiosize;
18895b699e3Smiod 	struct sisfb *fb;
18995b699e3Smiod 	int console;
19095b699e3Smiod 
19195b699e3Smiod 	if (pci_mapreg_map(pa, PCI_MAPREG_START, PCI_MAPREG_TYPE_MEM,
19295b699e3Smiod 	    BUS_SPACE_MAP_LINEAR, &fbt, &fbh, NULL, &fbsize, 0) != 0) {
19395b699e3Smiod 		printf(": can't map frame buffer\n");
19495b699e3Smiod 		return;
19595b699e3Smiod 	}
19695b699e3Smiod 
19795b699e3Smiod 	if (pci_mapreg_map(pa, PCI_MAPREG_START + 4, PCI_MAPREG_TYPE_MEM,
19895b699e3Smiod 	    0, &mmiot, &mmioh, NULL, &mmiosize, 0) != 0) {
19995b699e3Smiod 		printf(": can't map mmio area\n");
20095b699e3Smiod 		goto fail1;
20195b699e3Smiod 	}
20295b699e3Smiod 
20395b699e3Smiod 	if (pci_mapreg_map(pa, PCI_MAPREG_START + 8, PCI_MAPREG_TYPE_IO,
20495b699e3Smiod 	    0, &iot, &ioh, NULL, NULL, 0) != 0) {
20595b699e3Smiod 		printf(": can't map registers\n");
20695b699e3Smiod 		goto fail2;
20795b699e3Smiod 	}
20895b699e3Smiod 
20995b699e3Smiod 	console = sisfbcn.ri.ri_hw != NULL;
21095b699e3Smiod 
21195b699e3Smiod 	if (console)
21295b699e3Smiod 		fb = &sisfbcn;
21395b699e3Smiod 	else
21495b699e3Smiod 		fb = &sc->sc_fb_store;
21595b699e3Smiod 
21695b699e3Smiod 	fb->sc = sc;
21795b699e3Smiod 	fb->fbt = fbt;
21895b699e3Smiod 	fb->fbh = fbh;
21995b699e3Smiod 	fb->mmiot = mmiot;
22095b699e3Smiod 	fb->mmioh = mmioh;
22195b699e3Smiod 	fb->iot = iot;
22295b699e3Smiod 	fb->ioh = ioh;
22395b699e3Smiod 	sc->sc_fb = fb;
22495b699e3Smiod 
22595b699e3Smiod 	if (!console) {
22695b699e3Smiod 		if (sisfb_setup(fb) != 0) {
22795b699e3Smiod 			printf(": can't setup frame buffer\n");
22895b699e3Smiod 			return;
22995b699e3Smiod 		}
23095b699e3Smiod 	}
23195b699e3Smiod 
232f64f6f3bSfcambus 	printf(": %dx%d, %dbpp\n",
23395b699e3Smiod 	    fb->ri.ri_width, fb->ri.ri_height, fb->ri.ri_depth);
23495b699e3Smiod 
23595b699e3Smiod 	sc->sc_scrlist[0] = &fb->wsd;
23695b699e3Smiod 	sc->sc_wsl.nscreens = 1;
23795b699e3Smiod 	sc->sc_wsl.screens = (const struct wsscreen_descr **)sc->sc_scrlist;
23895b699e3Smiod 
23995b699e3Smiod 	waa.console = console;
24095b699e3Smiod 	waa.scrdata = &sc->sc_wsl;
24195b699e3Smiod 	waa.accessops = &sisfb_accessops;
24295b699e3Smiod 	waa.accesscookie = sc;
24395b699e3Smiod 	waa.defaultscreens = 0;
24495b699e3Smiod 
24595b699e3Smiod 	config_found((struct device *)sc, &waa, wsemuldisplaydevprint);
24695b699e3Smiod 	return;
24795b699e3Smiod 
24895b699e3Smiod fail2:
24995b699e3Smiod 	bus_space_unmap(mmiot, mmioh, mmiosize);
25095b699e3Smiod fail1:
25195b699e3Smiod 	bus_space_unmap(fbt, fbh, fbsize);
25295b699e3Smiod }
25395b699e3Smiod 
25495b699e3Smiod /*
25595b699e3Smiod  * wsdisplay accesops
25695b699e3Smiod  */
25795b699e3Smiod 
25895b699e3Smiod int
sisfb_alloc_screen(void * v,const struct wsscreen_descr * type,void ** cookiep,int * curxp,int * curyp,uint32_t * attrp)25995b699e3Smiod sisfb_alloc_screen(void *v, const struct wsscreen_descr *type, void **cookiep,
260e0c3e559Sjsg     int *curxp, int *curyp, uint32_t *attrp)
26195b699e3Smiod {
26295b699e3Smiod 	struct sisfb_softc *sc = (struct sisfb_softc *)v;
26395b699e3Smiod 	struct rasops_info *ri = &sc->sc_fb->ri;
26495b699e3Smiod 
26595b699e3Smiod 	if (sc->sc_nscr > 0)
26695b699e3Smiod 		return ENOMEM;
26795b699e3Smiod 
26895b699e3Smiod 	*cookiep = ri;
26995b699e3Smiod 	*curxp = *curyp = 0;
270fc223b23Sjsg 	ri->ri_ops.pack_attr(ri, 0, 0, 0, attrp);
27195b699e3Smiod 	sc->sc_nscr++;
27295b699e3Smiod 
27395b699e3Smiod 	return 0;
27495b699e3Smiod }
27595b699e3Smiod 
27695b699e3Smiod void
sisfb_free_screen(void * v,void * cookie)27795b699e3Smiod sisfb_free_screen(void *v, void *cookie)
27895b699e3Smiod {
27995b699e3Smiod 	struct sisfb_softc *sc = (struct sisfb_softc *)v;
28095b699e3Smiod 
28195b699e3Smiod 	sc->sc_nscr--;
28295b699e3Smiod }
28395b699e3Smiod 
28495b699e3Smiod int
sisfb_ioctl(void * v,u_long cmd,caddr_t data,int flags,struct proc * p)28595b699e3Smiod sisfb_ioctl(void *v, u_long cmd, caddr_t data, int flags, struct proc *p)
28695b699e3Smiod {
28795b699e3Smiod 	struct sisfb_softc *sc = (struct sisfb_softc *)v;
28895b699e3Smiod 	struct sisfb *fb = sc->sc_fb;
28995b699e3Smiod 	struct rasops_info *ri = &fb->ri;
29095b699e3Smiod 	struct wsdisplay_cmap *cm;
29195b699e3Smiod 	struct wsdisplay_fbinfo *wdf;
29295b699e3Smiod 	int rc;
29395b699e3Smiod 
29495b699e3Smiod 	switch (cmd) {
29595b699e3Smiod 	case WSDISPLAYIO_GTYPE:
29695b699e3Smiod 		*(uint *)data = WSDISPLAY_TYPE_SISFB;
29795b699e3Smiod 		break;
29895b699e3Smiod 	case WSDISPLAYIO_GINFO:
29995b699e3Smiod 		wdf = (struct wsdisplay_fbinfo *)data;
30095b699e3Smiod 		wdf->width = ri->ri_width;
30195b699e3Smiod 		wdf->height = ri->ri_height;
30295b699e3Smiod 		wdf->depth = ri->ri_depth;
303*63294167Skettenis 		wdf->stride = ri->ri_stride;
304*63294167Skettenis 		wdf->offset = 0;
30595b699e3Smiod 		wdf->cmsize = 256;
30695b699e3Smiod 		break;
30795b699e3Smiod 	case WSDISPLAYIO_LINEBYTES:
30895b699e3Smiod 		*(uint *)data = ri->ri_stride;
30995b699e3Smiod 		break;
31095b699e3Smiod 	case WSDISPLAYIO_GETCMAP:
31195b699e3Smiod 		cm = (struct wsdisplay_cmap *)data;
31295b699e3Smiod 		rc = sisfb_getcmap(fb->cmap, cm);
31395b699e3Smiod 		if (rc != 0)
31495b699e3Smiod 			return rc;
31595b699e3Smiod 		break;
31695b699e3Smiod 	case WSDISPLAYIO_PUTCMAP:
31795b699e3Smiod 		cm = (struct wsdisplay_cmap *)data;
31895b699e3Smiod 		rc = sisfb_putcmap(fb->cmap, cm);
31995b699e3Smiod 		if (rc != 0)
32095b699e3Smiod 			return rc;
32195b699e3Smiod 		if (ri->ri_depth == 8)
32295b699e3Smiod 			sisfb_loadcmap(fb, cm->index, cm->count);
32395b699e3Smiod 		break;
32495b699e3Smiod 	default:
32595b699e3Smiod 		return -1;
32695b699e3Smiod 	}
32795b699e3Smiod 
32895b699e3Smiod 	return 0;
32995b699e3Smiod }
33095b699e3Smiod 
33195b699e3Smiod int
sisfb_show_screen(void * v,void * cookie,int waitok,void (* cb)(void *,int,int),void * cbarg)33295b699e3Smiod sisfb_show_screen(void *v, void *cookie, int waitok,
33395b699e3Smiod     void (*cb)(void *, int, int), void *cbarg)
33495b699e3Smiod {
33595b699e3Smiod 	return 0;
33695b699e3Smiod }
33795b699e3Smiod 
33895b699e3Smiod paddr_t
sisfb_mmap(void * v,off_t offset,int prot)33995b699e3Smiod sisfb_mmap(void *v, off_t offset, int prot)
34095b699e3Smiod {
34195b699e3Smiod 	struct sisfb_softc *sc = (struct sisfb_softc *)v;
34295b699e3Smiod 	struct rasops_info *ri = &sc->sc_fb->ri;
34395b699e3Smiod 
34495b699e3Smiod 	if ((offset & PAGE_MASK) != 0)
34595b699e3Smiod 		return -1;
34695b699e3Smiod 
34795b699e3Smiod 	if (offset < 0 || offset >= ri->ri_stride * ri->ri_height)
34895b699e3Smiod 		return -1;
34995b699e3Smiod 
35095b699e3Smiod 	/*
35195b699e3Smiod 	 * Don't allow mmap if the frame buffer area is not page aligned.
35295b699e3Smiod 	 * XXX we should reprogram it to a page aligned boundary at attach
35395b699e3Smiod 	 * XXX time if this isn't the case.
35495b699e3Smiod 	 */
35595b699e3Smiod 	if (((paddr_t)ri->ri_bits & PAGE_MASK) != 0)
35695b699e3Smiod 		return -1;
35795b699e3Smiod 
35897c974cdSmiod 	return XKPHYS_TO_PHYS((paddr_t)ri->ri_bits) + offset;
35995b699e3Smiod }
36095b699e3Smiod 
36183275742Smiod int
sisfb_load_font(void * v,void * emulcookie,struct wsdisplay_font * font)36283275742Smiod sisfb_load_font(void *v, void *emulcookie, struct wsdisplay_font *font)
36383275742Smiod {
36483275742Smiod 	struct sisfb_softc *sc = (struct sisfb_softc *)v;
36583275742Smiod 	struct rasops_info *ri = &sc->sc_fb->ri;
36683275742Smiod 
36783275742Smiod 	return rasops_load_font(ri, emulcookie, font);
36883275742Smiod }
36983275742Smiod 
37083275742Smiod int
sisfb_list_font(void * v,struct wsdisplay_font * font)37183275742Smiod sisfb_list_font(void *v, struct wsdisplay_font *font)
37283275742Smiod {
37383275742Smiod 	struct sisfb_softc *sc = (struct sisfb_softc *)v;
37483275742Smiod 	struct rasops_info *ri = &sc->sc_fb->ri;
37583275742Smiod 
37683275742Smiod 	return rasops_list_font(ri, font);
37783275742Smiod }
37883275742Smiod 
37995b699e3Smiod /*
38095b699e3Smiod  * Frame buffer initialization.
38195b699e3Smiod  */
38295b699e3Smiod 
38395b699e3Smiod int
sisfb_setup(struct sisfb * fb)38495b699e3Smiod sisfb_setup(struct sisfb *fb)
38595b699e3Smiod {
38695b699e3Smiod 	struct rasops_info *ri;
38795b699e3Smiod 	uint width, height, bpp;
38895b699e3Smiod 	bus_size_t fbaddr;
38995b699e3Smiod 	uint tmp;
39095b699e3Smiod 
39195b699e3Smiod 	/*
39295b699e3Smiod 	 * Unlock access to extended registers.
39395b699e3Smiod 	 */
39495b699e3Smiod 
39595b699e3Smiod 	sisfb_seq_write(fb, 0x05, 0x86);
39695b699e3Smiod 
39795b699e3Smiod 	/*
39895b699e3Smiod 	 * Try and figure out display settings.
39995b699e3Smiod 	 */
40095b699e3Smiod 
40195b699e3Smiod 	height = sisfb_crtc_read(fb, CRTC_VDE);
40295b699e3Smiod 	tmp = sisfb_crtc_read(fb, CRTC_OVERFLL);
40395b699e3Smiod 	if (ISSET(tmp, 1 << 1))
40495b699e3Smiod 		height |= 1 << 8;
40595b699e3Smiod 	if (ISSET(tmp, 1 << 6))
40695b699e3Smiod 		height |= 1 << 9;
40795b699e3Smiod 	tmp = sisfb_seq_read(fb, 0x0a);
40895b699e3Smiod 	if (ISSET(tmp, 1 << 1))
40995b699e3Smiod 		height |= 1 << 10;
41095b699e3Smiod 	height++;
41195b699e3Smiod 
41295b699e3Smiod 	width = sisfb_crtc_read(fb, CRTC_HDISPLE);
41395b699e3Smiod 	tmp = sisfb_seq_read(fb, 0x0b);
41495b699e3Smiod 	if (ISSET(tmp, 1 << 2))
41595b699e3Smiod 		width |= 1 << 8;
41695b699e3Smiod 	if (ISSET(tmp, 1 << 3))
41795b699e3Smiod 		width |= 1 << 9;
41895b699e3Smiod 	width++;
41995b699e3Smiod 	width <<= 3;
42095b699e3Smiod 
42195b699e3Smiod 	fbaddr = sisfb_crtc_read(fb, CRTC_STARTADRL) |
42295b699e3Smiod 	    (sisfb_crtc_read(fb, CRTC_STARTADRH) << 8) |
42395b699e3Smiod 	    (sisfb_seq_read(fb, 0x0d) << 16) |
42495b699e3Smiod 	    ((sisfb_seq_read(fb, 0x37) & 0x03) << 24);
42595b699e3Smiod 	fbaddr <<= 2;
42695b699e3Smiod #ifdef SIS_DEBUG
42795b699e3Smiod 	printf("FBADDR %08x\n", fbaddr);
42895b699e3Smiod #endif
42995b699e3Smiod 
43095b699e3Smiod 	tmp = sisfb_seq_read(fb, 0x06);
43195b699e3Smiod 	switch (tmp & 0x1c) {
43295b699e3Smiod 	case 0x00:
43395b699e3Smiod 		bpp = 8;
43495b699e3Smiod 		break;
43595b699e3Smiod 	case 0x04:
43695b699e3Smiod 		bpp = 15;
43795b699e3Smiod 		break;
43895b699e3Smiod 	case 0x08:
43995b699e3Smiod 		bpp = 16;
44095b699e3Smiod 		break;
44195b699e3Smiod 	case 0x10:
44295b699e3Smiod 		bpp = 32;
44395b699e3Smiod 		break;
44495b699e3Smiod 	default:
44595b699e3Smiod 		return EINVAL;
44695b699e3Smiod 	}
44795b699e3Smiod 
44895b699e3Smiod 	ri = &fb->ri;
44995b699e3Smiod 	ri->ri_width = width;
45095b699e3Smiod 	ri->ri_height = height;
45195b699e3Smiod 	ri->ri_depth = bpp;
45295b699e3Smiod 	ri->ri_stride = (ri->ri_width * ri->ri_depth) / 8;
45395b699e3Smiod 	ri->ri_flg = RI_CENTER | RI_CLEAR | RI_FULLCLEAR;
45495b699e3Smiod 	ri->ri_bits = (void *)(bus_space_vaddr(fb->fbt, fb->fbh) + fbaddr);
45595b699e3Smiod 	ri->ri_hw = fb;
45695b699e3Smiod 
45795b699e3Smiod #ifdef __MIPSEL__
45895b699e3Smiod 	/* swap B and R */
45995b699e3Smiod 	switch (bpp) {
46095b699e3Smiod 	case 15:
46195b699e3Smiod 		ri->ri_rnum = 5;
46295b699e3Smiod 		ri->ri_rpos = 10;
46395b699e3Smiod 		ri->ri_gnum = 5;
46495b699e3Smiod 		ri->ri_gpos = 5;
46595b699e3Smiod 		ri->ri_bnum = 5;
46695b699e3Smiod 		ri->ri_bpos = 0;
46795b699e3Smiod 		break;
46895b699e3Smiod 	case 16:
46995b699e3Smiod 		ri->ri_rnum = 5;
47095b699e3Smiod 		ri->ri_rpos = 11;
47195b699e3Smiod 		ri->ri_gnum = 6;
47295b699e3Smiod 		ri->ri_gpos = 5;
47395b699e3Smiod 		ri->ri_bnum = 5;
47495b699e3Smiod 		ri->ri_bpos = 0;
47595b699e3Smiod 		break;
47695b699e3Smiod 	}
47795b699e3Smiod #endif
47895b699e3Smiod 
47995b699e3Smiod 	bcopy(rasops_cmap, fb->cmap, sizeof(fb->cmap));
48095b699e3Smiod 	if (bpp == 8)
48195b699e3Smiod 		sisfb_loadcmap(fb, 0, 256);
48295b699e3Smiod 
48395b699e3Smiod 	rasops_init(ri, 160, 160);
48495b699e3Smiod 
48595b699e3Smiod 	strlcpy(fb->wsd.name, "std", sizeof(fb->wsd.name));
48695b699e3Smiod 	fb->wsd.ncols = ri->ri_cols;
48795b699e3Smiod 	fb->wsd.nrows = ri->ri_rows;
48895b699e3Smiod 	fb->wsd.textops = &ri->ri_ops;
48995b699e3Smiod 	fb->wsd.fontwidth = ri->ri_font->fontwidth;
49095b699e3Smiod 	fb->wsd.fontheight = ri->ri_font->fontheight;
49195b699e3Smiod 	fb->wsd.capabilities = ri->ri_caps;
49295b699e3Smiod 
49395b699e3Smiod 	return 0;
49495b699e3Smiod }
49595b699e3Smiod 
49695b699e3Smiod /*
49795b699e3Smiod  * Colormap handling routines.
49895b699e3Smiod  */
49995b699e3Smiod 
50095b699e3Smiod void
sisfb_loadcmap(struct sisfb * fb,int baseidx,int count)50195b699e3Smiod sisfb_loadcmap(struct sisfb *fb, int baseidx, int count)
50295b699e3Smiod {
50395b699e3Smiod 	uint8_t *cmap = fb->cmap + baseidx * 3;
50495b699e3Smiod 
50595b699e3Smiod 	bus_space_write_1(fb->iot, fb->ioh, DAC_ADDR, baseidx);
50695b699e3Smiod 	while (count-- != 0) {
50795b699e3Smiod 		bus_space_write_1(fb->iot, fb->ioh, DAC_DATA, *cmap++ >> 2);
50895b699e3Smiod 		bus_space_write_1(fb->iot, fb->ioh, DAC_DATA, *cmap++ >> 2);
50995b699e3Smiod 		bus_space_write_1(fb->iot, fb->ioh, DAC_DATA, *cmap++ >> 2);
51095b699e3Smiod 	}
51195b699e3Smiod }
51295b699e3Smiod 
51395b699e3Smiod int
sisfb_getcmap(uint8_t * cmap,struct wsdisplay_cmap * cm)51495b699e3Smiod sisfb_getcmap(uint8_t *cmap, struct wsdisplay_cmap *cm)
51595b699e3Smiod {
51695b699e3Smiod 	uint index = cm->index, count = cm->count, i;
51795b699e3Smiod 	uint8_t ramp[256], *dst, *src;
51895b699e3Smiod 	int rc;
51995b699e3Smiod 
52095b699e3Smiod 	if (index >= 256 || count > 256 - index)
52195b699e3Smiod 		return EINVAL;
52295b699e3Smiod 
52395b699e3Smiod 	index *= 3;
52495b699e3Smiod 
52595b699e3Smiod 	src = cmap + index;
52695b699e3Smiod 	dst = ramp;
52795b699e3Smiod 	for (i = 0; i < count; i++)
52895b699e3Smiod 		*dst++ = *src, src += 3;
52995b699e3Smiod 	rc = copyout(ramp, cm->red, count);
53095b699e3Smiod 	if (rc != 0)
53195b699e3Smiod 		return rc;
53295b699e3Smiod 
53395b699e3Smiod 	src = cmap + index + 1;
53495b699e3Smiod 	dst = ramp;
53595b699e3Smiod 	for (i = 0; i < count; i++)
53695b699e3Smiod 		*dst++ = *src, src += 3;
53795b699e3Smiod 	rc = copyout(ramp, cm->green, count);
53895b699e3Smiod 	if (rc != 0)
53995b699e3Smiod 		return rc;
54095b699e3Smiod 
54195b699e3Smiod 	src = cmap + index + 2;
54295b699e3Smiod 	dst = ramp;
54395b699e3Smiod 	for (i = 0; i < count; i++)
54495b699e3Smiod 		*dst++ = *src, src += 3;
54595b699e3Smiod 	rc = copyout(ramp, cm->blue, count);
54695b699e3Smiod 	if (rc != 0)
54795b699e3Smiod 		return rc;
54895b699e3Smiod 
54995b699e3Smiod 	return 0;
55095b699e3Smiod }
55195b699e3Smiod 
55295b699e3Smiod int
sisfb_putcmap(uint8_t * cmap,struct wsdisplay_cmap * cm)55395b699e3Smiod sisfb_putcmap(uint8_t *cmap, struct wsdisplay_cmap *cm)
55495b699e3Smiod {
55595b699e3Smiod 	uint index = cm->index, count = cm->count, i;
55695b699e3Smiod 	uint8_t ramp[256], *dst, *src;
55795b699e3Smiod 	int rc;
55895b699e3Smiod 
55995b699e3Smiod 	if (index >= 256 || count > 256 - index)
56095b699e3Smiod 		return EINVAL;
56195b699e3Smiod 
56295b699e3Smiod 	index *= 3;
56395b699e3Smiod 
56495b699e3Smiod 	rc = copyin(cm->red, ramp, count);
56595b699e3Smiod 	if (rc != 0)
56695b699e3Smiod 		return rc;
56795b699e3Smiod 	dst = cmap + index;
56895b699e3Smiod 	src = ramp;
56995b699e3Smiod 	for (i = 0; i < count; i++)
57095b699e3Smiod 		*dst = *src++, dst += 3;
57195b699e3Smiod 
57295b699e3Smiod 	rc = copyin(cm->green, ramp, count);
57395b699e3Smiod 	if (rc != 0)
57495b699e3Smiod 		return rc;
57595b699e3Smiod 	dst = cmap + index + 1;
57695b699e3Smiod 	src = ramp;
57795b699e3Smiod 	for (i = 0; i < count; i++)
57895b699e3Smiod 		*dst = *src++, dst += 3;
57995b699e3Smiod 
58095b699e3Smiod 	rc = copyin(cm->blue, ramp, count);
58195b699e3Smiod 	if (rc != 0)
58295b699e3Smiod 		return rc;
58395b699e3Smiod 	dst = cmap + index + 2;
58495b699e3Smiod 	src = ramp;
58595b699e3Smiod 	for (i = 0; i < count; i++)
58695b699e3Smiod 		*dst = *src++, dst += 3;
58795b699e3Smiod 
58895b699e3Smiod 	return 0;
58995b699e3Smiod }
59095b699e3Smiod 
59195b699e3Smiod /*
59295b699e3Smiod  * Early console code
59395b699e3Smiod  */
59495b699e3Smiod 
59595b699e3Smiod int sisfb_cnattach(bus_space_tag_t, bus_space_tag_t, pcitag_t, pcireg_t);
59695b699e3Smiod 
59795b699e3Smiod int
sisfb_cnattach(bus_space_tag_t memt,bus_space_tag_t iot,pcitag_t tag,pcireg_t id)59895b699e3Smiod sisfb_cnattach(bus_space_tag_t memt, bus_space_tag_t iot, pcitag_t tag,
59995b699e3Smiod     pcireg_t id)
60095b699e3Smiod {
601e0c3e559Sjsg 	uint32_t defattr;
60295b699e3Smiod 	struct rasops_info *ri;
60395b699e3Smiod 	pcireg_t bar;
60495b699e3Smiod 	int rc;
60595b699e3Smiod 
60695b699e3Smiod 	/* filter out unrecognized devices */
60795b699e3Smiod 	switch (id) {
60895b699e3Smiod 	default:
60995b699e3Smiod 		return ENODEV;
61095b699e3Smiod 	case PCI_ID_CODE(PCI_VENDOR_SIS, PCI_PRODUCT_SIS_315PRO_VGA):
61195b699e3Smiod 		break;
61295b699e3Smiod 	}
61395b699e3Smiod 
61495b699e3Smiod 	bar = pci_conf_read_early(tag, PCI_MAPREG_START);
61595b699e3Smiod 	if (PCI_MAPREG_TYPE(bar) != PCI_MAPREG_TYPE_MEM)
61695b699e3Smiod 		return EINVAL;
61795b699e3Smiod 	sisfbcn.fbt = memt;
61895b699e3Smiod 	rc = bus_space_map(memt, PCI_MAPREG_MEM_ADDR(bar), 1 /* XXX */,
61995b699e3Smiod 	    BUS_SPACE_MAP_LINEAR, &sisfbcn.fbh);
62095b699e3Smiod 	if (rc != 0)
62195b699e3Smiod 		return rc;
62295b699e3Smiod 
62395b699e3Smiod 	bar = pci_conf_read_early(tag, PCI_MAPREG_START + 4);
62495b699e3Smiod 	if (PCI_MAPREG_TYPE(bar) != PCI_MAPREG_TYPE_MEM)
62595b699e3Smiod 		return EINVAL;
62695b699e3Smiod 	sisfbcn.mmiot = memt;
62795b699e3Smiod 	rc = bus_space_map(memt, PCI_MAPREG_MEM_ADDR(bar), 1 /* XXX */,
62895b699e3Smiod 	    BUS_SPACE_MAP_LINEAR, &sisfbcn.mmioh);
62995b699e3Smiod 	if (rc != 0)
63095b699e3Smiod 		return rc;
63195b699e3Smiod 
63295b699e3Smiod 	bar = pci_conf_read_early(tag, PCI_MAPREG_START + 8);
63395b699e3Smiod 	if (PCI_MAPREG_TYPE(bar) != PCI_MAPREG_TYPE_IO)
63495b699e3Smiod 		return EINVAL;
63595b699e3Smiod 	sisfbcn.iot = iot;
63695b699e3Smiod 	rc = bus_space_map(iot, PCI_MAPREG_MEM_ADDR(bar), 1 /* XXX */,
63795b699e3Smiod 	    0, &sisfbcn.ioh);
63895b699e3Smiod 	if (rc != 0)
63995b699e3Smiod 		return rc;
64095b699e3Smiod 
64195b699e3Smiod 	rc = sisfb_setup(&sisfbcn);
64295b699e3Smiod 	if (rc != 0)
64395b699e3Smiod 		return rc;
64495b699e3Smiod 
64595b699e3Smiod 	ri = &sisfbcn.ri;
646fc223b23Sjsg 	ri->ri_ops.pack_attr(ri, 0, 0, 0, &defattr);
64795b699e3Smiod 	wsdisplay_cnattach(&sisfbcn.wsd, ri, 0, 0, defattr);
64895b699e3Smiod 
64995b699e3Smiod 	return 0;
65095b699e3Smiod }
651