xref: /netbsd-src/sys/arch/pmax/ibus/pm.c (revision c986eee995756fff9d43af223e969be2818e7abd)
1*c986eee9Stsutsui /*	$NetBSD: pm.c,v 1.19 2023/02/11 18:30:45 tsutsui Exp $	*/
29aa9b8ebSad 
39aa9b8ebSad /*-
49aa9b8ebSad  * Copyright (c) 2002, 2003 The NetBSD Foundation, Inc.
59aa9b8ebSad  * All rights reserved.
69aa9b8ebSad  *
79aa9b8ebSad  * This code is derived from software contributed to The NetBSD Foundation
89aa9b8ebSad  * by Andrew Doran.
99aa9b8ebSad  *
109aa9b8ebSad  * Redistribution and use in source and binary forms, with or without
119aa9b8ebSad  * modification, are permitted provided that the following conditions
129aa9b8ebSad  * are met:
139aa9b8ebSad  * 1. Redistributions of source code must retain the above copyright
149aa9b8ebSad  *    notice, this list of conditions and the following disclaimer.
159aa9b8ebSad  * 2. Redistributions in binary form must reproduce the above copyright
169aa9b8ebSad  *    notice, this list of conditions and the following disclaimer in the
179aa9b8ebSad  *    documentation and/or other materials provided with the distribution.
189aa9b8ebSad  *
199aa9b8ebSad  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
209aa9b8ebSad  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
219aa9b8ebSad  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
229aa9b8ebSad  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
239aa9b8ebSad  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
249aa9b8ebSad  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
259aa9b8ebSad  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
269aa9b8ebSad  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
279aa9b8ebSad  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
289aa9b8ebSad  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
299aa9b8ebSad  * POSSIBILITY OF SUCH DAMAGE.
309aa9b8ebSad  */
319aa9b8ebSad 
329aa9b8ebSad #include <sys/cdefs.h>
33*c986eee9Stsutsui __KERNEL_RCSID(0, "$NetBSD: pm.c,v 1.19 2023/02/11 18:30:45 tsutsui Exp $");
349aa9b8ebSad 
359aa9b8ebSad #include <sys/param.h>
369aa9b8ebSad #include <sys/buf.h>
37a0640b2bSmatt #include <sys/bus.h>
38a0640b2bSmatt #include <sys/device.h>
399aa9b8ebSad #include <sys/ioctl.h>
40a0640b2bSmatt #include <sys/intr.h>
41a0640b2bSmatt #include <sys/kernel.h>
42a0640b2bSmatt #include <sys/systm.h>
439aa9b8ebSad 
449aa9b8ebSad #include <dev/wscons/wsconsio.h>
459aa9b8ebSad #include <dev/wscons/wsdisplayvar.h>
469aa9b8ebSad #include <dev/rasops/rasops.h>
479aa9b8ebSad #include <dev/wsfont/wsfont.h>
489aa9b8ebSad 
49*c986eee9Stsutsui #include <dev/ic/dc503reg.h>
50*c986eee9Stsutsui 
519aa9b8ebSad #include <pmax/pmax/kn01.h>
529aa9b8ebSad 
539aa9b8ebSad #include <pmax/ibus/ibusvar.h>
549aa9b8ebSad #include <pmax/ibus/pmreg.h>
559aa9b8ebSad 
569aa9b8ebSad #include <uvm/uvm_extern.h>
579aa9b8ebSad 
589aa9b8ebSad #define	CURSOR_MAX_SIZE	16
599aa9b8ebSad 
609aa9b8ebSad struct hwcmap256 {
619aa9b8ebSad 	uint8_t r[256];
629aa9b8ebSad 	uint8_t g[256];
639aa9b8ebSad 	uint8_t b[256];
649aa9b8ebSad };
659aa9b8ebSad 
669aa9b8ebSad struct hwcursor64 {
679aa9b8ebSad 	struct wsdisplay_curpos cc_pos;
689aa9b8ebSad 	struct wsdisplay_curpos cc_hot;
699aa9b8ebSad 	struct wsdisplay_curpos cc_size;
709aa9b8ebSad 	uint8_t cc_color[6];
719aa9b8ebSad 
729aa9b8ebSad 	/*
739aa9b8ebSad 	 * Max cursor size is 16x16.  The X server pads bitmap scanlines to
749aa9b8ebSad 	 * a word boundary.  We take the easy route and waste some space.
759aa9b8ebSad 	 */
769aa9b8ebSad 	u_short cc_image[32 + 32];
779aa9b8ebSad };
789aa9b8ebSad 
799aa9b8ebSad struct pm_softc {
80b4c2594fStsutsui 	device_t		sc_dev;
819aa9b8ebSad 	size_t			sc_cmap_size;
829aa9b8ebSad 	size_t			sc_fb_size;
839aa9b8ebSad 	int			sc_type;
849aa9b8ebSad 	int			sc_blanked;
859aa9b8ebSad 	int			sc_curenb;
869aa9b8ebSad 	int			sc_changed;
879aa9b8ebSad 	int			sc_nscreens;
889aa9b8ebSad 	struct hwcursor64	sc_cursor;
899aa9b8ebSad 	struct hwcmap256	sc_cmap;
909aa9b8ebSad };
919aa9b8ebSad #define	WSDISPLAY_CMAP_DOLUT	0x20
929aa9b8ebSad 
93b4c2594fStsutsui int	pm_match(device_t, cfdata_t, void *);
94b4c2594fStsutsui void	pm_attach(device_t, device_t, void *);
958322a83cSflxd int	pm_check_vfb(void);
9653524e44Schristos int	pm_ioctl(void *, void *, u_long, void *, int, struct lwp *);
977a51d4ddSjmmv paddr_t	pm_mmap(void *, void *, off_t, int);
989aa9b8ebSad int	pm_alloc_screen(void *, const struct wsscreen_descr *,
999aa9b8ebSad 				void **, int *, int *, long *);
1009aa9b8ebSad void	pm_free_screen(void *, void *);
1019aa9b8ebSad int	pm_show_screen(void *, void *, int,
1029aa9b8ebSad 			       void (*) (void *, int, int), void *);
1039aa9b8ebSad void	pm_cursor_off(void);
1049aa9b8ebSad void	pm_cursor_on(struct pm_softc *);
1059aa9b8ebSad int	pm_cnattach(void);
1069aa9b8ebSad void	pm_common_init(void);
1079aa9b8ebSad int	pm_flush(struct pm_softc *);
1089aa9b8ebSad int	pm_get_cmap(struct pm_softc *, struct wsdisplay_cmap *);
1099aa9b8ebSad int	pm_set_cmap(struct pm_softc *, struct wsdisplay_cmap *);
1109aa9b8ebSad int	pm_set_cursor(struct pm_softc *, struct wsdisplay_cursor *);
1119aa9b8ebSad int	pm_get_cursor(struct pm_softc *, struct wsdisplay_cursor *);
1129aa9b8ebSad void	pm_set_curpos(struct pm_softc *, struct wsdisplay_curpos *);
1139aa9b8ebSad void	pm_init_cmap(struct pm_softc *);
1149aa9b8ebSad 
115b4c2594fStsutsui CFATTACH_DECL_NEW(pm, sizeof(struct pm_softc),
1169aa9b8ebSad    pm_match, pm_attach, NULL, NULL);
1179aa9b8ebSad 
1189aa9b8ebSad struct rasops_info pm_ri;
1199aa9b8ebSad 
1209aa9b8ebSad struct wsscreen_descr pm_stdscreen = {
1219aa9b8ebSad 	"std", 0, 0,
1229aa9b8ebSad 	0, /* textops */
1239aa9b8ebSad 	0, 0,
1249aa9b8ebSad 	WSSCREEN_REVERSE
1259aa9b8ebSad };
1269aa9b8ebSad 
1279aa9b8ebSad const struct wsscreen_descr *_pm_scrlist[] = {
1289aa9b8ebSad 	&pm_stdscreen,
1299aa9b8ebSad };
1309aa9b8ebSad 
1319aa9b8ebSad const struct wsscreen_list pm_screenlist = {
1329aa9b8ebSad 	sizeof(_pm_scrlist) / sizeof(struct wsscreen_descr *), _pm_scrlist
1339aa9b8ebSad };
1349aa9b8ebSad 
1359aa9b8ebSad const struct wsdisplay_accessops pm_accessops = {
1369aa9b8ebSad 	pm_ioctl,
1379aa9b8ebSad 	pm_mmap,
1389aa9b8ebSad 	pm_alloc_screen,
1399aa9b8ebSad 	pm_free_screen,
1409aa9b8ebSad 	pm_show_screen,
1419aa9b8ebSad 	0 /* load_font */
1429aa9b8ebSad };
1439aa9b8ebSad 
1449aa9b8ebSad u_int	pm_creg;
1459aa9b8ebSad 
1469aa9b8ebSad int
pm_match(device_t parent,cfdata_t cf,void * aux)147b4c2594fStsutsui pm_match(device_t parent, cfdata_t cf, void *aux)
1489aa9b8ebSad {
1499aa9b8ebSad 	struct ibus_attach_args *ia;
15053524e44Schristos 	void *pmaddr;
1519aa9b8ebSad 
1529aa9b8ebSad 	ia = aux;
15353524e44Schristos 	pmaddr = (void *)ia->ia_addr;
1549aa9b8ebSad 
1559aa9b8ebSad 	if (strcmp(ia->ia_name, "pm") != 0)
1569aa9b8ebSad 		return (0);
1579aa9b8ebSad 
1589aa9b8ebSad 	if (badaddr(pmaddr, 4))
1599aa9b8ebSad 		return (0);
1609aa9b8ebSad 
1619aa9b8ebSad 	return (1);
1629aa9b8ebSad }
1639aa9b8ebSad 
1649aa9b8ebSad void
pm_attach(device_t parent,device_t self,void * aux)165b4c2594fStsutsui pm_attach(device_t parent, device_t self, void *aux)
1669aa9b8ebSad {
1679aa9b8ebSad 	struct pm_softc *sc;
1689aa9b8ebSad 	struct rasops_info *ri;
1699aa9b8ebSad 	struct wsemuldisplaydev_attach_args waa;
1709aa9b8ebSad 	int console;
1719aa9b8ebSad 
172b4c2594fStsutsui 	sc = device_private(self);
173b4c2594fStsutsui 	sc->sc_dev = self;
1749aa9b8ebSad 	ri = &pm_ri;
1759aa9b8ebSad 	console = (ri->ri_bits != NULL);
1769aa9b8ebSad 
1776f304e5aStsutsui 	if (console) {
1789aa9b8ebSad 		sc->sc_nscreens = 1;
1796f304e5aStsutsui 		ri->ri_flg &= ~RI_NO_AUTO;
1808322a83cSflxd 	} else if (!pm_check_vfb()) {
1818322a83cSflxd 		printf(": VFB01/VFB02 frame buffer option not found\n");
1828322a83cSflxd 		return;
1836f304e5aStsutsui 	} else
1849aa9b8ebSad 		pm_common_init();
1859aa9b8ebSad 
1869aa9b8ebSad 	printf(": %dx%d, %dbpp\n", ri->ri_width, ri->ri_height, ri->ri_depth);
1879aa9b8ebSad 
1889aa9b8ebSad 	pm_init_cmap(sc);
1899aa9b8ebSad 
1909aa9b8ebSad 	sc->sc_blanked = 0;
1919aa9b8ebSad 	sc->sc_curenb = 0;
1929aa9b8ebSad 
1939aa9b8ebSad 	waa.console = console;
1949aa9b8ebSad 	waa.scrdata = &pm_screenlist;
1959aa9b8ebSad 	waa.accessops = &pm_accessops;
1969aa9b8ebSad 	waa.accesscookie = sc;
1979aa9b8ebSad 
198c7fb772bSthorpej 	config_found(self, &waa, wsemuldisplaydevprint, CFARGS_NONE);
1999aa9b8ebSad }
2009aa9b8ebSad 
2018322a83cSflxd int
pm_check_vfb(void)2028322a83cSflxd pm_check_vfb(void)
2038322a83cSflxd {
2048322a83cSflxd 	int *mem;
2058322a83cSflxd 	const int magic = 0xcafebabe;
2068322a83cSflxd 
2078322a83cSflxd 	mem = (void *)MIPS_PHYS_TO_KSEG1(KN01_PHYS_FBUF_START);
2088322a83cSflxd 
2098322a83cSflxd 	*mem = magic;
2108322a83cSflxd 	wbflush();
2118322a83cSflxd 	if (*mem != magic)
2128322a83cSflxd 		return 0;
2138322a83cSflxd 
2148322a83cSflxd 	*mem = ~magic;
2158322a83cSflxd 	wbflush();
2168322a83cSflxd 	if (*mem != ~magic)
2178322a83cSflxd 		return 0;
2188322a83cSflxd 
2198322a83cSflxd 	return 1;
2208322a83cSflxd }
2218322a83cSflxd 
2229aa9b8ebSad void
pm_init_cmap(struct pm_softc * sc)2239aa9b8ebSad pm_init_cmap(struct pm_softc *sc)
2249aa9b8ebSad {
2259aa9b8ebSad 	struct hwcmap256 *cm;
2269aa9b8ebSad 	struct rasops_info *ri;
2279aa9b8ebSad 	const uint8_t *p;
2289aa9b8ebSad 	int index;
2299aa9b8ebSad 
2309aa9b8ebSad 	cm = &sc->sc_cmap;
2319aa9b8ebSad 	ri = &pm_ri;
2329aa9b8ebSad 
2339aa9b8ebSad 	if (ri->ri_depth == 8) {
2349aa9b8ebSad 		p = rasops_cmap;
2359aa9b8ebSad 		for (index = 0; index < 256; index++, p += 3) {
2369aa9b8ebSad 			cm->r[index] = p[0];
2379aa9b8ebSad 			cm->g[index] = p[1];
2389aa9b8ebSad 			cm->b[index] = p[2];
2399aa9b8ebSad 		}
2409aa9b8ebSad 
2419aa9b8ebSad 		sc->sc_type = WSDISPLAY_TYPE_PM_COLOR;
2429aa9b8ebSad 		sc->sc_cmap_size = 256;
2439aa9b8ebSad 		sc->sc_fb_size = 0x100000;
2449aa9b8ebSad 	} else {
2459aa9b8ebSad 		cm->r[0] = 0x00;
2469aa9b8ebSad 		cm->g[0] = 0x00;
2479aa9b8ebSad 		cm->b[0] = 0x00;
2489aa9b8ebSad 
2499aa9b8ebSad 		cm->r[1] = 0x00;
2509aa9b8ebSad 		cm->g[1] = 0xff;
2519aa9b8ebSad 		cm->b[1] = 0x00;
2529aa9b8ebSad 
2539aa9b8ebSad 		sc->sc_type = WSDISPLAY_TYPE_PM_MONO;
2549aa9b8ebSad 		sc->sc_cmap_size = 2;
2559aa9b8ebSad 		sc->sc_fb_size = 0x40000;
2569aa9b8ebSad 	}
2579aa9b8ebSad }
2589aa9b8ebSad 
2599aa9b8ebSad void
pm_common_init(void)2609aa9b8ebSad pm_common_init(void)
2619aa9b8ebSad {
2629aa9b8ebSad 	struct rasops_info *ri;
2639aa9b8ebSad 	int cookie, bior, i;
264*c986eee9Stsutsui 	struct dc503reg *pcc;
2659aa9b8ebSad 	VDACRegs *vdac;
2669aa9b8ebSad 	uint16_t kn01csr;
2679aa9b8ebSad 
2689aa9b8ebSad 	kn01csr = *(volatile uint16_t *)MIPS_PHYS_TO_KSEG1(KN01_SYS_CSR);
2699aa9b8ebSad 	pcc = (void *)MIPS_PHYS_TO_KSEG1(KN01_SYS_PCC);
2709aa9b8ebSad 	vdac = (void *)MIPS_PHYS_TO_KSEG1(KN01_SYS_VDAC);
2719aa9b8ebSad 	ri = &pm_ri;
2729aa9b8ebSad 
2739aa9b8ebSad 	ri->ri_flg = RI_CENTER;
2746f304e5aStsutsui 	if (ri->ri_bits == NULL)
2756f304e5aStsutsui 		ri->ri_flg |= RI_NO_AUTO;
2769aa9b8ebSad 	ri->ri_depth = ((kn01csr & KN01_CSR_MONO) != 0 ? 1 : 8);
2779aa9b8ebSad 	ri->ri_width = 1024;
2789aa9b8ebSad 	ri->ri_height = 864;
2799aa9b8ebSad 	ri->ri_stride = (ri->ri_depth == 8 ? 1024 : 2048 / 8);
2809aa9b8ebSad 	ri->ri_bits = (void *)MIPS_PHYS_TO_KSEG1(KN01_PHYS_FBUF_START);
2819aa9b8ebSad 
2829aa9b8ebSad 	/*
2839aa9b8ebSad 	 * Clear the screen.
2849aa9b8ebSad 	 */
2859aa9b8ebSad 	memset(ri->ri_bits, 0, ri->ri_stride * ri->ri_height);
2869aa9b8ebSad 
2879aa9b8ebSad 	/*
2889aa9b8ebSad 	 * Get a font to use.
2899aa9b8ebSad 	 */
2909aa9b8ebSad 	bior = (ri->ri_depth == 8 ? WSDISPLAY_FONTORDER_L2R :
2919aa9b8ebSad 	    WSDISPLAY_FONTORDER_R2L);
2929aa9b8ebSad 
2939aa9b8ebSad 	wsfont_init();
2949aa9b8ebSad 	if (ri->ri_depth == 8)
2959aa9b8ebSad 		cookie = wsfont_find(NULL, 12, 0, 0, bior,
296ec24591fSmacallan 		    WSDISPLAY_FONTORDER_L2R, WSFONT_FIND_BITMAP);
2979aa9b8ebSad 	else
2989aa9b8ebSad 		cookie = wsfont_find(NULL, 8, 0, 0, bior,
299ec24591fSmacallan 		    WSDISPLAY_FONTORDER_L2R, WSFONT_FIND_BITMAP);
3009aa9b8ebSad 	if (cookie <= 0)
3019aa9b8ebSad 		cookie = wsfont_find(NULL, 0, 0, 0, bior,
302ec24591fSmacallan 		    WSDISPLAY_FONTORDER_L2R, WSFONT_FIND_BITMAP);
3039aa9b8ebSad 	if (cookie <= 0) {
3049aa9b8ebSad 		printf("pm: font table is empty\n");
3059aa9b8ebSad 		return;
3069aa9b8ebSad 	}
3079aa9b8ebSad 
3089aa9b8ebSad 	if (wsfont_lock(cookie, &ri->ri_font)) {
3099aa9b8ebSad 		printf("pm: couldn't lock font\n");
3109aa9b8ebSad 		return;
3119aa9b8ebSad 	}
3129aa9b8ebSad 	ri->ri_wsfcookie = cookie;
3139aa9b8ebSad 
3149aa9b8ebSad 	/*
3159aa9b8ebSad 	 * Set up the raster operations set.
3169aa9b8ebSad 	 */
3179aa9b8ebSad 	rasops_init(ri, 1000, 1000);
3189aa9b8ebSad 
3199aa9b8ebSad 	pm_stdscreen.nrows = ri->ri_rows;
3209aa9b8ebSad 	pm_stdscreen.ncols = ri->ri_cols;
3219aa9b8ebSad 	pm_stdscreen.textops = &ri->ri_ops;
3229aa9b8ebSad 	pm_stdscreen.capabilities = ri->ri_caps;
3239aa9b8ebSad 
3249aa9b8ebSad 	/*
325c56890eeSmsaitoh 	 * Initialize the VDAC.
3269aa9b8ebSad 	 */
3279aa9b8ebSad 	*(uint8_t *)MIPS_PHYS_TO_KSEG1(KN01_PHYS_COLMASK_START) = 0xff;
3289aa9b8ebSad 	wbflush();
3299aa9b8ebSad 
3309aa9b8ebSad 	vdac->overWA = 0x04; wbflush();
3319aa9b8ebSad 	vdac->over = 0x00; wbflush();
3329aa9b8ebSad 	vdac->over = 0x00; wbflush();
3339aa9b8ebSad 	vdac->over = 0x00; wbflush();
3349aa9b8ebSad 	vdac->overWA = 0x08; wbflush();
3359aa9b8ebSad 	vdac->over = 0x00; wbflush();
3369aa9b8ebSad 	vdac->over = 0x00; wbflush();
3379aa9b8ebSad 	vdac->over = 0x7f; wbflush();
3389aa9b8ebSad 	vdac->overWA = 0x0c; wbflush();
3399aa9b8ebSad 	vdac->over = 0xff; wbflush();
3409aa9b8ebSad 	vdac->over = 0xff; wbflush();
3419aa9b8ebSad 	vdac->over = 0xff; wbflush();
3429aa9b8ebSad 
3439aa9b8ebSad 	/*
3449aa9b8ebSad 	 * Set in the initial colormap.
3459aa9b8ebSad 	 */
3469aa9b8ebSad 	if (ri->ri_depth == 8) {
3479aa9b8ebSad 		vdac->mapWA = 0;
3489aa9b8ebSad 		wbflush();
3499aa9b8ebSad 
3509aa9b8ebSad 		for (i = 0; i < 256 * 3; i += 3) {
3519aa9b8ebSad 			vdac->map = rasops_cmap[i];
3529aa9b8ebSad 			wbflush();
3539aa9b8ebSad 			vdac->map = rasops_cmap[i + 1];
3549aa9b8ebSad 			wbflush();
3559aa9b8ebSad 			vdac->map = rasops_cmap[i + 2];
3569aa9b8ebSad 			wbflush();
3579aa9b8ebSad 		}
3589aa9b8ebSad 	} else {
3599aa9b8ebSad 		vdac->mapWA = 0;
3609aa9b8ebSad 		wbflush();
3619aa9b8ebSad 
3629aa9b8ebSad 		for (i = 0; i < 256; i++) {
3639aa9b8ebSad 			vdac->map = 0x00;
3649aa9b8ebSad 			wbflush();
3659aa9b8ebSad 			vdac->map = (i < 128 ? 0x00 : 0xff);
3669aa9b8ebSad 			wbflush();
3679aa9b8ebSad 			vdac->map = 0x00;
3689aa9b8ebSad 			wbflush();
3699aa9b8ebSad 		}
3709aa9b8ebSad 	}
3719aa9b8ebSad 
3729aa9b8ebSad 	/*
3739aa9b8ebSad 	 * Turn off the hardware cursor sprite for text mode.
3749aa9b8ebSad 	 */
375*c986eee9Stsutsui 	pcc->cmdr = PCCCMD_FOPB | PCCCMD_VBHI;
3769aa9b8ebSad 	wbflush();
3779aa9b8ebSad 	pm_creg = 0;
3789aa9b8ebSad 	pm_cursor_off();
3799aa9b8ebSad }
3809aa9b8ebSad 
3819aa9b8ebSad void
pm_cursor_off(void)3829aa9b8ebSad pm_cursor_off(void)
3839aa9b8ebSad {
384*c986eee9Stsutsui 	struct dc503reg *pcc;
3859aa9b8ebSad 
3869aa9b8ebSad 	pcc = (void *)MIPS_PHYS_TO_KSEG1(KN01_SYS_PCC);
387*c986eee9Stsutsui 	pcc->cmdr = (pm_creg &= ~(PCCCMD_ENPA | PCCCMD_ENPB));
3889aa9b8ebSad 	wbflush();
3899aa9b8ebSad }
3909aa9b8ebSad 
3919aa9b8ebSad void
pm_cursor_on(struct pm_softc * sc)3929aa9b8ebSad pm_cursor_on(struct pm_softc *sc)
3939aa9b8ebSad {
394*c986eee9Stsutsui 	struct dc503reg *pcc;
3959aa9b8ebSad 
3969aa9b8ebSad 	if (sc->sc_curenb) {
3979aa9b8ebSad 		pcc = (void *)MIPS_PHYS_TO_KSEG1(KN01_SYS_PCC);
398*c986eee9Stsutsui 		pcc->cmdr = (pm_creg |= (PCCCMD_ENPA | PCCCMD_ENPB));
3999aa9b8ebSad 		wbflush();
4009aa9b8ebSad 	}
4019aa9b8ebSad }
4029aa9b8ebSad 
4039aa9b8ebSad int
pm_ioctl(void * v,void * vs,u_long cmd,void * data,int flag,struct lwp * l)40453524e44Schristos pm_ioctl(void *v, void *vs, u_long cmd, void *data, int flag, struct lwp *l)
4059aa9b8ebSad {
4069aa9b8ebSad 	struct pm_softc *sc;
4079aa9b8ebSad 	struct rasops_info *ri;
4089aa9b8ebSad 	int turnoff, rv, i;
409*c986eee9Stsutsui 	struct dc503reg *pcc;
4109aa9b8ebSad 	VDACRegs *vdac;
4119aa9b8ebSad 
4129aa9b8ebSad 	sc = v;
4139aa9b8ebSad 	ri = &pm_ri;
4149aa9b8ebSad 	rv = 0;
4159aa9b8ebSad 	pcc = (void *)MIPS_PHYS_TO_KSEG1(KN01_SYS_PCC);
4169aa9b8ebSad 	vdac = (void *)MIPS_PHYS_TO_KSEG1(KN01_SYS_VDAC);
4179aa9b8ebSad 
4189aa9b8ebSad 	switch (cmd) {
4199aa9b8ebSad 	case WSDISPLAYIO_GTYPE:
4209aa9b8ebSad 		*(u_int *)data = sc->sc_type;
4219aa9b8ebSad 		break;
4229aa9b8ebSad 
4239aa9b8ebSad 	case WSDISPLAYIO_SMODE:
4249aa9b8ebSad 		if (*(u_int *)data == WSDISPLAYIO_MODE_EMUL) {
4259aa9b8ebSad 			pm_cursor_off();
4269aa9b8ebSad 			pm_init_cmap(sc);
4279aa9b8ebSad 			memset(ri->ri_bits, 0, ri->ri_stride * ri->ri_height);
4289aa9b8ebSad 			sc->sc_curenb = 0;
4299aa9b8ebSad 			sc->sc_changed |= WSDISPLAY_CMAP_DOLUT;
4309aa9b8ebSad 		}
4319aa9b8ebSad 		break;
4329aa9b8ebSad 
4339aa9b8ebSad 	case WSDISPLAYIO_GINFO:
4349aa9b8ebSad #define	wsd_fbip ((struct wsdisplay_fbinfo *)data)
4359aa9b8ebSad 		wsd_fbip->height = ri->ri_height;
4369aa9b8ebSad 		wsd_fbip->width = ri->ri_width;
4379aa9b8ebSad 		wsd_fbip->depth = ri->ri_depth;
4389aa9b8ebSad 		wsd_fbip->cmsize = sc->sc_cmap_size;
4399aa9b8ebSad #undef fbt
4409aa9b8ebSad 		break;
4419aa9b8ebSad 
4429aa9b8ebSad 	case WSDISPLAYIO_GETCMAP:
4439aa9b8ebSad 		rv = pm_get_cmap(sc, (struct wsdisplay_cmap *)data);
4449aa9b8ebSad 		break;
4459aa9b8ebSad 
4469aa9b8ebSad 	case WSDISPLAYIO_PUTCMAP:
4479aa9b8ebSad 		rv = pm_set_cmap(sc, (struct wsdisplay_cmap *)data);
4489aa9b8ebSad 		break;
4499aa9b8ebSad 
4509aa9b8ebSad 	case WSDISPLAYIO_SVIDEO:
4519aa9b8ebSad 		turnoff = (*(int *)data == WSDISPLAYIO_VIDEO_OFF);
4529aa9b8ebSad 		if ((sc->sc_blanked == 0) ^ turnoff) {
4539aa9b8ebSad 			sc->sc_blanked = turnoff;
4549aa9b8ebSad 			if (turnoff == 0) {
4559aa9b8ebSad 				pcc->cmdr =
456*c986eee9Stsutsui 				    (pm_creg &= ~(PCCCMD_FOPA | PCCCMD_FOPB));
4579aa9b8ebSad 				wbflush();
4589aa9b8ebSad 				pm_cursor_on(sc);
4599aa9b8ebSad 				sc->sc_changed |= WSDISPLAY_CURSOR_DOCMAP;
4609aa9b8ebSad 			} else {
4619aa9b8ebSad 				pm_cursor_off();
4629aa9b8ebSad 				pcc->cmdr =
463*c986eee9Stsutsui 				    (pm_creg |= (PCCCMD_FOPA | PCCCMD_FOPB));
4649aa9b8ebSad 				wbflush();
4659aa9b8ebSad 				vdac->overWA = 0x0c;
4669aa9b8ebSad 				wbflush();
4679aa9b8ebSad 				for (i = 0; i < 3; i++) {
4689aa9b8ebSad 					vdac->over = 0;
4699aa9b8ebSad 					wbflush();
4709aa9b8ebSad 				}
4719aa9b8ebSad 			}
4729aa9b8ebSad 		}
4739aa9b8ebSad 		break;
4749aa9b8ebSad 
4759aa9b8ebSad 	case WSDISPLAYIO_GVIDEO:
4769aa9b8ebSad 		*(u_int *)data = (sc->sc_blanked ?
4779aa9b8ebSad 		    WSDISPLAYIO_VIDEO_OFF : WSDISPLAYIO_VIDEO_ON);
4789aa9b8ebSad 		break;
4799aa9b8ebSad 
4809aa9b8ebSad 	case WSDISPLAYIO_GCURPOS:
4819aa9b8ebSad 		*(struct wsdisplay_curpos *)data = sc->sc_cursor.cc_pos;
4829aa9b8ebSad 		break;
4839aa9b8ebSad 
4849aa9b8ebSad 	case WSDISPLAYIO_SCURPOS:
4859aa9b8ebSad 		pm_set_curpos(sc, (struct wsdisplay_curpos *)data);
4869aa9b8ebSad 		sc->sc_changed |= WSDISPLAY_CURSOR_DOPOS;
4879aa9b8ebSad 		break;
4889aa9b8ebSad 
4899aa9b8ebSad 	case WSDISPLAYIO_GCURMAX:
4909aa9b8ebSad 		((struct wsdisplay_curpos *)data)->x =
4919aa9b8ebSad 		((struct wsdisplay_curpos *)data)->y = CURSOR_MAX_SIZE;
4929aa9b8ebSad 		break;
4939aa9b8ebSad 
4949aa9b8ebSad 	case WSDISPLAYIO_GCURSOR:
4959aa9b8ebSad 		rv = pm_get_cursor(sc, (struct wsdisplay_cursor *)data);
4969aa9b8ebSad 		break;
4979aa9b8ebSad 
4989aa9b8ebSad 	case WSDISPLAYIO_SCURSOR:
4999aa9b8ebSad 		rv = pm_set_cursor(sc, (struct wsdisplay_cursor *)data);
5009aa9b8ebSad 		break;
5019aa9b8ebSad 
5029aa9b8ebSad 	default:
5039aa9b8ebSad 		rv = ENOTTY;
5049aa9b8ebSad 		break;
5059aa9b8ebSad 	}
5069aa9b8ebSad 
5079aa9b8ebSad 	pm_flush(sc);
5089aa9b8ebSad 	return (rv);
5099aa9b8ebSad }
5109aa9b8ebSad 
5119aa9b8ebSad paddr_t
pm_mmap(void * v,void * vs,off_t offset,int prot)5127a51d4ddSjmmv pm_mmap(void *v, void *vs, off_t offset, int prot)
5139aa9b8ebSad {
5149aa9b8ebSad 	struct pm_softc *sc;
5159aa9b8ebSad 
5169aa9b8ebSad 	sc = v;
5179aa9b8ebSad 
5189aa9b8ebSad 	if (offset >= sc->sc_fb_size || offset < 0)
5199aa9b8ebSad 		return (-1);
5209aa9b8ebSad 
5219aa9b8ebSad 	return (mips_btop(KN01_PHYS_FBUF_START + offset));
5229aa9b8ebSad }
5239aa9b8ebSad 
5249aa9b8ebSad int
pm_alloc_screen(void * v,const struct wsscreen_descr * type,void ** cookiep,int * curxp,int * curyp,long * attrp)5259aa9b8ebSad pm_alloc_screen(void *v, const struct wsscreen_descr *type, void **cookiep,
5269aa9b8ebSad 		int *curxp, int *curyp, long *attrp)
5279aa9b8ebSad {
5289aa9b8ebSad 	struct pm_softc *sc;
5299aa9b8ebSad 	struct rasops_info *ri;
5309aa9b8ebSad 	long defattr;
5319aa9b8ebSad 
5329aa9b8ebSad 	sc = v;
5339aa9b8ebSad 	ri = &pm_ri;
5349aa9b8ebSad 
5359aa9b8ebSad 	if (sc->sc_nscreens > 0)
5369aa9b8ebSad 		return (ENOMEM);
5379aa9b8ebSad 
5389aa9b8ebSad 	*cookiep = ri;	 /* one and only for now */
5399aa9b8ebSad 	*curxp = 0;
5409aa9b8ebSad 	*curyp = 0;
5419aa9b8ebSad 	(*ri->ri_ops.allocattr)(ri, 0, 0, 0, &defattr);
5429aa9b8ebSad 	*attrp = defattr;
5439aa9b8ebSad 	sc->sc_nscreens++;
5449aa9b8ebSad 	return (0);
5459aa9b8ebSad }
5469aa9b8ebSad 
5479aa9b8ebSad void
pm_free_screen(void * v,void * cookie)5489aa9b8ebSad pm_free_screen(void *v, void *cookie)
5499aa9b8ebSad {
5509aa9b8ebSad 
5519aa9b8ebSad 	panic("pm_free_screen: console");
5529aa9b8ebSad }
5539aa9b8ebSad 
5549aa9b8ebSad int
pm_show_screen(void * v,void * cookie,int waitok,void (* cb)(void *,int,int),void * cbarg)5559aa9b8ebSad pm_show_screen(void *v, void *cookie, int waitok,
5569aa9b8ebSad 	       void (*cb)(void *, int, int), void *cbarg)
5579aa9b8ebSad {
5589aa9b8ebSad 
5599aa9b8ebSad 	return (0);
5609aa9b8ebSad }
5619aa9b8ebSad 
5629aa9b8ebSad /* EXPORT */ int
pm_cnattach(void)5639aa9b8ebSad pm_cnattach(void)
5649aa9b8ebSad {
5659aa9b8ebSad 	struct rasops_info *ri;
5669aa9b8ebSad 	long defattr;
5679aa9b8ebSad 
5689aa9b8ebSad 	ri = &pm_ri;
5699aa9b8ebSad 
5709aa9b8ebSad 	pm_common_init();
5719aa9b8ebSad 	(*ri->ri_ops.allocattr)(ri, 0, 0, 0, &defattr);
5729aa9b8ebSad 	wsdisplay_cnattach(&pm_stdscreen, ri, 0, 0, defattr);
5739aa9b8ebSad 	return (1);
5749aa9b8ebSad }
5759aa9b8ebSad 
5769aa9b8ebSad int
pm_flush(struct pm_softc * sc)5779aa9b8ebSad pm_flush(struct pm_softc *sc)
5789aa9b8ebSad {
5799aa9b8ebSad 	VDACRegs *vdac;
580*c986eee9Stsutsui 	struct dc503reg *pcc;
5819aa9b8ebSad 	uint8_t *cp;
5829aa9b8ebSad 	int v, i, x, y;
5839aa9b8ebSad 	u_short *p, *pe;
5849aa9b8ebSad 	struct hwcmap256 *cm;
5859aa9b8ebSad 
5869aa9b8ebSad 	if (sc->sc_changed == 0)
5879aa9b8ebSad 		return (1);
5889aa9b8ebSad 
5899aa9b8ebSad 	vdac = (void *)MIPS_PHYS_TO_KSEG1(KN01_SYS_VDAC);
5909aa9b8ebSad 	pcc = (void *)MIPS_PHYS_TO_KSEG1(KN01_SYS_PCC);
5919aa9b8ebSad 	v = sc->sc_changed;
5929aa9b8ebSad 
5939aa9b8ebSad 	if ((v & WSDISPLAY_CURSOR_DOCUR) != 0) {
5949aa9b8ebSad 		if (sc->sc_curenb)
5959aa9b8ebSad 			pm_cursor_on(sc);
5969aa9b8ebSad 		else
5979aa9b8ebSad 			pm_cursor_off();
5989aa9b8ebSad 	}
5999aa9b8ebSad 
6009aa9b8ebSad 	if ((v & (WSDISPLAY_CURSOR_DOPOS | WSDISPLAY_CURSOR_DOHOT)) != 0) {
6019aa9b8ebSad 		x = sc->sc_cursor.cc_pos.x - sc->sc_cursor.cc_hot.x;
6029aa9b8ebSad 		y = sc->sc_cursor.cc_pos.y - sc->sc_cursor.cc_hot.y;
6039aa9b8ebSad 		pcc->xpos = x + PCC_X_OFFSET;
6049aa9b8ebSad 		pcc->ypos = y + PCC_Y_OFFSET;
6059aa9b8ebSad 		wbflush();
6069aa9b8ebSad 	}
6079aa9b8ebSad 	if ((v & WSDISPLAY_CURSOR_DOCMAP) != 0) {
6089aa9b8ebSad 		cp = sc->sc_cursor.cc_color;
6099aa9b8ebSad 
6109aa9b8ebSad 		vdac->overWA = 0x04;
6119aa9b8ebSad 		wbflush();
6129aa9b8ebSad 		for (i = 1; i < 6; i += 2) {
6139aa9b8ebSad 			vdac->over = cp[i];
6149aa9b8ebSad 			wbflush();
6159aa9b8ebSad 		}
6169aa9b8ebSad 
6179aa9b8ebSad 		vdac->overWA = 0x08;
6189aa9b8ebSad 		wbflush();
6199aa9b8ebSad 		vdac->over = 0x00;
6209aa9b8ebSad 		wbflush();
6219aa9b8ebSad 		vdac->over = 0x00;
6229aa9b8ebSad 		wbflush();
6239aa9b8ebSad 		vdac->over = 0x7f;
6249aa9b8ebSad 		wbflush();
6259aa9b8ebSad 
6269aa9b8ebSad 		vdac->overWA = 0x0c;
6279aa9b8ebSad 		wbflush();
6289aa9b8ebSad 		for (i = 0; i < 6; i += 2) {
6299aa9b8ebSad 			vdac->over = cp[i];
6309aa9b8ebSad 			wbflush();
6319aa9b8ebSad 		}
6329aa9b8ebSad 	}
6339aa9b8ebSad 	if ((v & WSDISPLAY_CURSOR_DOSHAPE) != 0) {
634*c986eee9Stsutsui 		pcc->cmdr = (pm_creg | PCCCMD_LODSA);
6359aa9b8ebSad 		wbflush();
6369aa9b8ebSad 
6379aa9b8ebSad 		p = sc->sc_cursor.cc_image;
6389aa9b8ebSad 		x = 0xffff >> (16 - sc->sc_cursor.cc_size.x);
6399aa9b8ebSad 		for (pe = p + 64; p < pe; p += 2) {
640*c986eee9Stsutsui 			pcc->load = *p & x;
6419aa9b8ebSad 			wbflush();
6429aa9b8ebSad 		}
6439aa9b8ebSad 
644*c986eee9Stsutsui 		pcc->cmdr = (pm_creg &= ~PCCCMD_LODSA);
6459aa9b8ebSad 		wbflush();
6469aa9b8ebSad 	}
6479aa9b8ebSad 
6489aa9b8ebSad 	if ((v & WSDISPLAY_CMAP_DOLUT) != 0) {
6499aa9b8ebSad 		cm = &sc->sc_cmap;
6509aa9b8ebSad 
6519aa9b8ebSad 		vdac->mapWA = 0;
6529aa9b8ebSad 		wbflush();
6539aa9b8ebSad 
6549aa9b8ebSad 		if (sc->sc_cmap_size == 2) {
6559aa9b8ebSad 			for (i = 0; i < 128; i++) {
6569aa9b8ebSad 				vdac->map = 0;
6579aa9b8ebSad 				wbflush();
6589aa9b8ebSad 				vdac->map = cm->g[0];
6599aa9b8ebSad 				wbflush();
6609aa9b8ebSad 				vdac->map = 0;
6619aa9b8ebSad 				wbflush();
6629aa9b8ebSad 			}
6639aa9b8ebSad 			for (; i < 256; i++) {
6649aa9b8ebSad 				vdac->map = 0;
6659aa9b8ebSad 				wbflush();
6669aa9b8ebSad 				vdac->map = cm->g[1];
6679aa9b8ebSad 				wbflush();
6689aa9b8ebSad 				vdac->map = 0;
6699aa9b8ebSad 				wbflush();
6709aa9b8ebSad 			}
6719aa9b8ebSad 		} else {
6729aa9b8ebSad 			for (i = 0; i < sc->sc_cmap_size; i++) {
6739aa9b8ebSad 				vdac->map = cm->r[i];
6749aa9b8ebSad 				wbflush();
6759aa9b8ebSad 				vdac->map = cm->g[i];
6769aa9b8ebSad 				wbflush();
6779aa9b8ebSad 				vdac->map = cm->b[i];
6789aa9b8ebSad 				wbflush();
6799aa9b8ebSad 			}
6809aa9b8ebSad 		}
6819aa9b8ebSad 	}
6829aa9b8ebSad 
6839aa9b8ebSad 	sc->sc_changed = 0;
6849aa9b8ebSad 	return (1);
6859aa9b8ebSad }
6869aa9b8ebSad 
6879aa9b8ebSad int
pm_get_cmap(struct pm_softc * sc,struct wsdisplay_cmap * p)6889aa9b8ebSad pm_get_cmap(struct pm_softc *sc, struct wsdisplay_cmap *p)
6899aa9b8ebSad {
6909aa9b8ebSad 	u_int index, count;
6919aa9b8ebSad 	int rv;
6929aa9b8ebSad 
6939aa9b8ebSad 	index = p->index;
6949aa9b8ebSad 	count = p->count;
6959aa9b8ebSad 
6964b70fd12Sspz 	if (index >= sc->sc_cmap_size || count > sc->sc_cmap_size - index)
6979aa9b8ebSad 		return (EINVAL);
6989aa9b8ebSad 
6999aa9b8ebSad 	if ((rv = copyout(&sc->sc_cmap.r[index], p->red, count)) != 0)
7009aa9b8ebSad 		return (rv);
7019aa9b8ebSad 	if ((rv = copyout(&sc->sc_cmap.g[index], p->green, count)) != 0)
7029aa9b8ebSad 		return (rv);
7039aa9b8ebSad 	return (copyout(&sc->sc_cmap.b[index], p->blue, count));
7049aa9b8ebSad }
7059aa9b8ebSad 
7069aa9b8ebSad int
pm_set_cmap(struct pm_softc * sc,struct wsdisplay_cmap * p)7079aa9b8ebSad pm_set_cmap(struct pm_softc *sc, struct wsdisplay_cmap *p)
7089aa9b8ebSad {
7099aa9b8ebSad 	u_int index, count;
7109aa9b8ebSad 	int rv;
7119aa9b8ebSad 
7129aa9b8ebSad 	index = p->index;
7139aa9b8ebSad 	count = p->count;
7149aa9b8ebSad 
7154b70fd12Sspz 	if (index >= sc->sc_cmap_size || count > sc->sc_cmap_size - index)
7169aa9b8ebSad 		return (EINVAL);
7179aa9b8ebSad 
7189aa9b8ebSad 	if ((rv = copyin(p->red, &sc->sc_cmap.r[index], count)) != 0)
7199aa9b8ebSad 		return (rv);
7209aa9b8ebSad 	if ((rv = copyin(p->green, &sc->sc_cmap.g[index], count)) != 0)
7219aa9b8ebSad 		return (rv);
7229aa9b8ebSad 	if ((rv = copyin(p->blue, &sc->sc_cmap.b[index], count)) != 0)
7239aa9b8ebSad 		return (rv);
7249aa9b8ebSad 	sc->sc_changed |= WSDISPLAY_CMAP_DOLUT;
7259aa9b8ebSad 	return (0);
7269aa9b8ebSad }
7279aa9b8ebSad 
7289aa9b8ebSad int
pm_set_cursor(struct pm_softc * sc,struct wsdisplay_cursor * p)7299aa9b8ebSad pm_set_cursor(struct pm_softc *sc, struct wsdisplay_cursor *p)
7309aa9b8ebSad {
7319aa9b8ebSad 	u_int v, index, count;
7329aa9b8ebSad 	struct hwcursor64 *cc;
7339aa9b8ebSad 	int rv;
7349aa9b8ebSad 
7359aa9b8ebSad 	v = p->which;
7369aa9b8ebSad 	cc = &sc->sc_cursor;
7379aa9b8ebSad 
7389aa9b8ebSad 	if ((v & WSDISPLAY_CURSOR_DOCUR) != 0)
7399aa9b8ebSad 		sc->sc_curenb = p->enable;
7409aa9b8ebSad 	if ((v & WSDISPLAY_CURSOR_DOPOS) != 0)
7419aa9b8ebSad 		pm_set_curpos(sc, &p->pos);
7429aa9b8ebSad 	if ((v & WSDISPLAY_CURSOR_DOHOT) != 0)
7439aa9b8ebSad 		cc->cc_hot = p->hot;
7449aa9b8ebSad 	if ((v & WSDISPLAY_CURSOR_DOCMAP) != 0) {
7459aa9b8ebSad 		index = p->cmap.index;
7469aa9b8ebSad 		count = p->cmap.count;
7478e9f762cSriastradh 		if (index >= 2 || count > 2 - index)
7489aa9b8ebSad 			return (EINVAL);
7499aa9b8ebSad 
7509aa9b8ebSad 		rv = copyin(p->cmap.red, &cc->cc_color[index], count);
7519aa9b8ebSad 		if (rv != 0)
7529aa9b8ebSad 			return (rv);
7539aa9b8ebSad 		rv = copyin(p->cmap.green, &cc->cc_color[index + 2], count);
7549aa9b8ebSad 		if (rv != 0)
7559aa9b8ebSad 			return (rv);
7569aa9b8ebSad 		rv = copyin(p->cmap.blue, &cc->cc_color[index + 4], count);
7579aa9b8ebSad 		if (rv != 0)
7589aa9b8ebSad 			return (rv);
7599aa9b8ebSad 	}
7609aa9b8ebSad 	if ((v & WSDISPLAY_CURSOR_DOSHAPE) != 0) {
7619aa9b8ebSad 		if (p->size.x > CURSOR_MAX_SIZE ||
7629aa9b8ebSad 		    p->size.y > CURSOR_MAX_SIZE)
7639aa9b8ebSad 			return (EINVAL);
7649aa9b8ebSad 
7659aa9b8ebSad 		cc->cc_size = p->size;
7669aa9b8ebSad 		memset(cc->cc_image, 0, sizeof(cc->cc_image));
7679aa9b8ebSad 		rv = copyin(p->image, cc->cc_image, p->size.y * 4);
7689aa9b8ebSad 		if (rv != 0)
7699aa9b8ebSad 			return (rv);
7709aa9b8ebSad 		rv = copyin(p->mask, cc->cc_image+32, p->size.y * 4);
7719aa9b8ebSad 		if (rv != 0)
7729aa9b8ebSad 			return (rv);
7739aa9b8ebSad 	}
7749aa9b8ebSad 
7759aa9b8ebSad 	sc->sc_changed |= v;
7769aa9b8ebSad 	return (0);
7779aa9b8ebSad }
7789aa9b8ebSad 
7799aa9b8ebSad int
pm_get_cursor(struct pm_softc * sc,struct wsdisplay_cursor * p)7809aa9b8ebSad pm_get_cursor(struct pm_softc *sc, struct wsdisplay_cursor *p)
7819aa9b8ebSad {
7829aa9b8ebSad 
7839aa9b8ebSad 	return (ENOTTY); /* XXX */
7849aa9b8ebSad }
7859aa9b8ebSad 
7869aa9b8ebSad void
pm_set_curpos(struct pm_softc * sc,struct wsdisplay_curpos * curpos)7879aa9b8ebSad pm_set_curpos(struct pm_softc *sc, struct wsdisplay_curpos *curpos)
7889aa9b8ebSad {
7899aa9b8ebSad 	struct rasops_info *ri;
7909aa9b8ebSad 	int x, y;
7919aa9b8ebSad 
7929aa9b8ebSad 	ri = &pm_ri;
7939aa9b8ebSad 	x = curpos->x;
7949aa9b8ebSad 	y = curpos->y;
7959aa9b8ebSad 
7969aa9b8ebSad 	if (y < 0)
7979aa9b8ebSad 		y = 0;
7989aa9b8ebSad 	else if (y > ri->ri_height)
7999aa9b8ebSad 		y = ri->ri_height;
8009aa9b8ebSad 	if (x < 0)
8019aa9b8ebSad 		x = 0;
8029aa9b8ebSad 	else if (x > ri->ri_width)
8039aa9b8ebSad 		x = ri->ri_width;
8049aa9b8ebSad 	sc->sc_cursor.cc_pos.x = x;
8059aa9b8ebSad 	sc->sc_cursor.cc_pos.y = y;
8069aa9b8ebSad }
807