gsfb.c revision 1.23 1 1.23 thorpej /* $NetBSD: gsfb.c,v 1.23 2021/04/24 23:36:45 thorpej Exp $ */
2 1.1 uch
3 1.1 uch /*-
4 1.1 uch * Copyright (c) 2001 The NetBSD Foundation, Inc.
5 1.1 uch * All rights reserved.
6 1.1 uch *
7 1.1 uch * This code is derived from software contributed to The NetBSD Foundation
8 1.1 uch * by UCHIYAMA Yasushi.
9 1.1 uch *
10 1.1 uch * Redistribution and use in source and binary forms, with or without
11 1.1 uch * modification, are permitted provided that the following conditions
12 1.1 uch * are met:
13 1.1 uch * 1. Redistributions of source code must retain the above copyright
14 1.1 uch * notice, this list of conditions and the following disclaimer.
15 1.1 uch * 2. Redistributions in binary form must reproduce the above copyright
16 1.1 uch * notice, this list of conditions and the following disclaimer in the
17 1.1 uch * documentation and/or other materials provided with the distribution.
18 1.1 uch *
19 1.1 uch * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20 1.1 uch * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21 1.1 uch * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22 1.1 uch * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23 1.1 uch * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24 1.1 uch * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25 1.1 uch * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26 1.1 uch * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27 1.1 uch * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28 1.1 uch * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29 1.1 uch * POSSIBILITY OF SUCH DAMAGE.
30 1.1 uch */
31 1.7 lukem
32 1.7 lukem #include <sys/cdefs.h>
33 1.23 thorpej __KERNEL_RCSID(0, "$NetBSD: gsfb.c,v 1.23 2021/04/24 23:36:45 thorpej Exp $");
34 1.1 uch
35 1.1 uch #include "debug_playstation2.h"
36 1.1 uch
37 1.1 uch #include <sys/param.h>
38 1.1 uch #include <sys/systm.h>
39 1.1 uch
40 1.1 uch #include <machine/autoconf.h>
41 1.1 uch
42 1.1 uch #include <dev/cons.h>
43 1.1 uch
44 1.1 uch #include <dev/wscons/wsconsio.h>
45 1.1 uch #include <dev/wscons/wsdisplayvar.h>
46 1.1 uch #include <dev/wscons/wscons_callbacks.h>
47 1.1 uch
48 1.1 uch #include <dev/wsfont/wsfont.h>
49 1.1 uch
50 1.1 uch #include <playstation2/ee/eevar.h>
51 1.1 uch #include <playstation2/ee/gsvar.h>
52 1.1 uch #include <playstation2/ee/gsreg.h>
53 1.1 uch #include <playstation2/ee/dmacvar.h>
54 1.1 uch #include <playstation2/ee/dmacreg.h>
55 1.1 uch
56 1.1 uch #ifdef DEBUG
57 1.1 uch #define STATIC
58 1.1 uch #else
59 1.1 uch #define STATIC static
60 1.1 uch #endif
61 1.1 uch
62 1.22 martin struct gsfb_softc {
63 1.22 martin device_t sc_dev;
64 1.22 martin const struct wsscreen_descr *sc_screen;
65 1.22 martin struct wsdisplay_font *sc_font;
66 1.22 martin bool sc_is_console;
67 1.22 martin };
68 1.22 martin
69 1.22 martin static int gsfb_is_console;
70 1.22 martin static struct gsfb_softc gsfb_console_softc;
71 1.1 uch
72 1.1 uch STATIC void gsfb_dma_kick(paddr_t, size_t);
73 1.1 uch STATIC void gsfb_font_expand_psmct32(const struct wsdisplay_font *, u_int,
74 1.1 uch long, u_int32_t *);
75 1.13 perry STATIC inline void gsfb_set_cursor_pos(u_int32_t *, int, int, int, int);
76 1.1 uch
77 1.1 uch #define ATTR_FG_GET(a) (((a )>> 24) & 0xf)
78 1.1 uch #define ATTR_BG_GET(a) (((a )>> 16) & 0xf)
79 1.1 uch #define ATTR_FG_SET(x) (((x) << 24) & 0x0f000000)
80 1.1 uch #define ATTR_BG_SET(x) (((x) << 16) & 0x000f0000)
81 1.1 uch
82 1.1 uch STATIC const u_int32_t gsfb_ansi_psmct32[] = {
83 1.1 uch 0x80000000, /* black */
84 1.1 uch 0x800000aa, /* red */
85 1.1 uch 0x8000aa00, /* green */
86 1.1 uch 0x8000aaaa, /* brown */
87 1.1 uch 0x80aa0000, /* blue */
88 1.1 uch 0x80aa00aa, /* magenta */
89 1.1 uch 0x80aaaa00, /* cyan */
90 1.1 uch 0x80aaaaaa, /* white */
91 1.1 uch 0x80000000, /* black */
92 1.1 uch 0x800000ff, /* red */
93 1.1 uch 0x8000ff00, /* green */
94 1.1 uch 0x8000ffff, /* brown */
95 1.1 uch 0x80ff0000, /* blue */
96 1.1 uch 0x80ff00ff, /* magenta */
97 1.1 uch 0x80ffff00, /* cyan */
98 1.1 uch 0x80ffffff, /* black */
99 1.1 uch };
100 1.1 uch
101 1.1 uch #define TRXPOS_DXY(f, x, y) \
102 1.1 uch ({ \
103 1.1 uch f[9] = ((x) & 0x000007ff) | (((y) << 16) & 0x07ff0000); \
104 1.1 uch })
105 1.1 uch
106 1.1 uch #define TRXPOS_SY_DY(f, sy, dy) \
107 1.1 uch ({ \
108 1.1 uch f[8] = (((sy) << 16) & 0x07ff0000); \
109 1.1 uch f[9] = (((dy) << 16) & 0x07ff0000); \
110 1.1 uch })
111 1.1 uch
112 1.1 uch #define TRXPOS_DXY_SXY(f, dx, dy, sx, sy) \
113 1.1 uch ({ \
114 1.1 uch f[8] = ((((sy) << 16) & 0x07ff0000) | ((sx) & 0x000007ff)); \
115 1.1 uch f[9] = ((((dy) << 16) & 0x07ff0000) | ((dx) & 0x000007ff)); \
116 1.1 uch })
117 1.1 uch
118 1.1 uch STATIC u_int32_t gsfb_scroll_cmd_640x16[] __attribute__((__aligned__(16))) = {
119 1.1 uch 0x00008004, 0x10000000, 0x0000000e, 0x00000000,
120 1.1 uch 0x000a0000, 0x000a0000, 0x00000050, 0x00000000,
121 1.1 uch 0x07ff0000, 0x07ff0000, 0x00000051, 0x00000000,
122 1.1 uch 0x00000280, 0x00000010, 0x00000052, 0x00000000,
123 1.1 uch 0x00000002, 0x00000000, 0x00000053, 0x00000000,
124 1.1 uch };
125 1.1 uch
126 1.1 uch STATIC u_int32_t gsfb_cursor_cmd[] __attribute__((__aligned__(16))) = {
127 1.1 uch 0x00008007, 0x10000000, 0x0000000e, 0x00000000,
128 1.1 uch 0x00000001, 0x00000000, 0x0000001a, 0x00000000,
129 1.1 uch 0x000000a4, 0x00000080, 0x00000042, 0x00000000,
130 1.1 uch 0x00000046, 0x00000000, 0x00000000, 0x00000000,
131 1.1 uch 0x80ffffff, 0x00000000, 0x00000001, 0x00000000,
132 1.1 uch 0x00000000, 0x00000000, 0x0000000d, 0x00000000,
133 1.1 uch 0x80ffffff, 0x00000000, 0x00000001, 0x00000000,
134 1.1 uch 0x00000000, 0x00000000, 0x00000005, 0x00000000,
135 1.1 uch };
136 1.1 uch
137 1.1 uch STATIC u_int32_t gsfb_copy_cmd_8x16[] __attribute__((__aligned__(16))) = {
138 1.1 uch 0x00008004, 0x10000000, 0x0000000e, 0x00000000,
139 1.1 uch 0x000a0000, 0x000a0000, 0x00000050, 0x00000000,
140 1.1 uch 0x07ff07ff, 0x07ff07ff, 0x00000051, 0x00000000,
141 1.1 uch 0x00000008, 0x00000010, 0x00000052, 0x00000000,
142 1.1 uch 0x00000002, 0x00000000, 0x00000053, 0x00000000,
143 1.1 uch };
144 1.1 uch
145 1.1 uch STATIC u_int32_t gsfb_init_cmd_640x480[] __attribute__((__aligned__(16))) = {
146 1.1 uch 0x00008008, 0x10000000, 0x0000000e, 0x00000000,
147 1.1 uch 0x000a0000, 0x00000000, 0x0000004c, 0x00000000,
148 1.1 uch 0x00000096, 0x00000000, 0x0000004e, 0x00000000,
149 1.1 uch 0x02800000, 0x01e00000, 0x00000040, 0x00000000,
150 1.1 uch 0x00000006, 0x00000000, 0x00000000, 0x00000000,
151 1.1 uch 0x80000000, 0x00000000, 0x00000001, 0x00000000,
152 1.1 uch 0x00000000, 0x00000000, 0x0000000d, 0x00000000,
153 1.1 uch 0x80000000, 0x00000000, 0x00000001, 0x00000000,
154 1.1 uch 0x1e002800, 0x00000000, 0x00000005, 0x00000000,
155 1.1 uch };
156 1.1 uch
157 1.1 uch STATIC u_int32_t gsfb_load_cmd_8x16_psmct32[(6 + 32) * 4]
158 1.1 uch __attribute__((__aligned__(16))) = {
159 1.1 uch /* GIF tag + GS command */
160 1.1 uch 0x00000004, 0x10000000, 0x0000000e, 0x00000000,
161 1.1 uch 0x00000000, 0x000a0000, 0x00000050, 0x00000000,
162 1.1 uch 0x00000000, 0x00000000, 0x00000051, 0x00000000,
163 1.1 uch 0x00000008, 0x00000016, 0x00000052, 0x00000000,
164 1.1 uch 0x00000000, 0x00000000, 0x00000053, 0x00000000,
165 1.1 uch 0x00008020, 0x08000000, 0x00000000, 0x00000000,
166 1.1 uch /* Load area */
167 1.1 uch #define FONT_SCRATCH_BASE (6 * 4)
168 1.1 uch };
169 1.1 uch
170 1.1 uch #ifdef GSFB_DEBUG_MONITOR
171 1.1 uch #include <machine/stdarg.h>
172 1.1 uch STATIC const struct _gsfb_debug_window {
173 1.1 uch int start, nrow, attr;
174 1.1 uch } _gsfb_debug_window[3] = {
175 1.1 uch { 24, 2 , ATTR_BG_SET(WSCOL_BROWN) | ATTR_FG_SET(WSCOL_BLUE) },
176 1.1 uch { 26, 2 , ATTR_BG_SET(WSCOL_CYAN) | ATTR_FG_SET(WSCOL_BLUE) },
177 1.1 uch { 28, 2 , ATTR_BG_SET(WSCOL_WHITE) | ATTR_FG_SET(WSCOL_BLUE) },
178 1.1 uch };
179 1.1 uch STATIC char _gsfb_debug_buf[80 * 2];
180 1.1 uch #endif /* GSFB_DEBUG_MONITOR */
181 1.1 uch
182 1.22 martin STATIC int gsfb_match(device_t, cfdata_t, void *);
183 1.22 martin STATIC void gsfb_attach(device_t, device_t, void *);
184 1.1 uch
185 1.22 martin CFATTACH_DECL_NEW(gsfb, sizeof(struct gsfb_softc),
186 1.6 thorpej gsfb_match, gsfb_attach, NULL, NULL);
187 1.1 uch
188 1.1 uch STATIC void gsfb_hwinit(void);
189 1.22 martin STATIC int gsfb_swinit(struct gsfb_softc*);
190 1.1 uch
191 1.1 uch /* console */
192 1.1 uch void gsfbcnprobe(struct consdev *);
193 1.1 uch void gsfbcninit(struct consdev *);
194 1.1 uch
195 1.1 uch /* emul ops */
196 1.1 uch STATIC void _gsfb_cursor(void *, int, int, int);
197 1.1 uch STATIC int _gsfb_mapchar(void *, int, unsigned int *);
198 1.1 uch STATIC void _gsfb_putchar(void *, int, int, u_int, long);
199 1.1 uch STATIC void _gsfb_copycols(void *, int, int, int, int);
200 1.1 uch STATIC void _gsfb_erasecols(void *, int, int, int, long);
201 1.1 uch STATIC void _gsfb_copyrows(void *, int, int, int);
202 1.1 uch STATIC void _gsfb_eraserows(void *, int, int, long);
203 1.4 junyoung STATIC int _gsfb_allocattr(void *, int, int, int, long *);
204 1.1 uch
205 1.1 uch /* access ops */
206 1.17 christos STATIC int _gsfb_ioctl(void *, void *, u_long, void *, int, struct lwp *);
207 1.16 jmmv STATIC paddr_t _gsfb_mmap(void *, void *, off_t, int);
208 1.1 uch STATIC int _gsfb_alloc_screen(void *, const struct wsscreen_descr *, void **,
209 1.1 uch int *, int *, long *);
210 1.1 uch STATIC void _gsfb_free_screen(void *, void *);
211 1.1 uch STATIC int _gsfb_show_screen(void *, void *, int, void (*)(void *, int, int),
212 1.1 uch void *);
213 1.1 uch STATIC void _gsfb_pollc(void *, int);
214 1.1 uch
215 1.1 uch /*
216 1.1 uch * wsdisplay attach args
217 1.1 uch * std: screen size 640 x 480, font size 8 x 16
218 1.1 uch */
219 1.1 uch #define GSFB_STD_SCREEN_WIDTH 640
220 1.1 uch #define GSFB_STD_SCREEN_HEIGHT 480
221 1.1 uch #define GSFB_STD_FONT_WIDTH 8
222 1.1 uch #define GSFB_STD_FONT_HEIGHT 16
223 1.1 uch const struct wsdisplay_emulops _gsfb_emulops = {
224 1.1 uch .cursor = _gsfb_cursor,
225 1.1 uch .mapchar = _gsfb_mapchar,
226 1.1 uch .putchar = _gsfb_putchar,
227 1.1 uch .copycols = _gsfb_copycols,
228 1.1 uch .erasecols = _gsfb_erasecols,
229 1.1 uch .copyrows = _gsfb_copyrows,
230 1.1 uch .eraserows = _gsfb_eraserows,
231 1.4 junyoung .allocattr = _gsfb_allocattr
232 1.1 uch };
233 1.1 uch
234 1.1 uch const struct wsscreen_descr _gsfb_std_screen = {
235 1.1 uch .name = "std",
236 1.1 uch .ncols = 80,
237 1.1 uch #ifdef GSFB_DEBUG_MONITOR
238 1.1 uch .nrows = 24,
239 1.1 uch #else
240 1.1 uch .nrows = 30,
241 1.1 uch #endif
242 1.1 uch .textops = &_gsfb_emulops,
243 1.1 uch .fontwidth = 8,
244 1.1 uch .fontheight = 16,
245 1.1 uch .capabilities = WSSCREEN_UNDERLINE | WSSCREEN_HILIT |
246 1.1 uch WSSCREEN_WSCOLORS
247 1.1 uch };
248 1.1 uch
249 1.1 uch const struct wsscreen_descr *_gsfb_screen_table[] = {
250 1.1 uch &_gsfb_std_screen,
251 1.1 uch };
252 1.1 uch
253 1.1 uch struct wsscreen_list _gsfb_screen_list = {
254 1.1 uch .nscreens = sizeof(_gsfb_screen_table) /
255 1.1 uch sizeof(_gsfb_screen_table[0]),
256 1.1 uch .screens = _gsfb_screen_table
257 1.1 uch };
258 1.1 uch
259 1.1 uch struct wsdisplay_accessops _gsfb_accessops = {
260 1.1 uch .ioctl = _gsfb_ioctl,
261 1.1 uch .mmap = _gsfb_mmap,
262 1.1 uch .alloc_screen = _gsfb_alloc_screen,
263 1.1 uch .free_screen = _gsfb_free_screen,
264 1.1 uch .show_screen = _gsfb_show_screen,
265 1.1 uch .load_font = 0,
266 1.1 uch .pollc = _gsfb_pollc
267 1.1 uch };
268 1.1 uch
269 1.1 uch int
270 1.22 martin gsfb_match(device_t parent, cfdata_t cf, void *aux)
271 1.1 uch {
272 1.1 uch extern struct cfdriver gsfb_cd;
273 1.1 uch struct mainbus_attach_args *ma = aux;
274 1.1 uch
275 1.1 uch if (strcmp(ma->ma_name, gsfb_cd.cd_name) != 0)
276 1.1 uch return (0);
277 1.1 uch
278 1.22 martin return 1;
279 1.1 uch }
280 1.1 uch
281 1.1 uch void
282 1.22 martin gsfb_attach(device_t parent, device_t self, void *aux)
283 1.1 uch {
284 1.1 uch struct wsemuldisplaydev_attach_args wa;
285 1.22 martin struct gsfb_softc *sc = device_private(self);
286 1.22 martin
287 1.22 martin if (gsfb_is_console) {
288 1.22 martin memcpy(sc, &gsfb_console_softc, sizeof(gsfb_console_softc));
289 1.22 martin sc->sc_is_console = true;
290 1.22 martin }
291 1.22 martin sc->sc_dev = self;
292 1.1 uch
293 1.22 martin if (!sc->sc_is_console && !gsfb_swinit(sc) != 0)
294 1.1 uch return;
295 1.1 uch
296 1.1 uch printf("\n");
297 1.1 uch
298 1.22 martin wa.console = sc->sc_is_console;
299 1.1 uch wa.scrdata = &_gsfb_screen_list;
300 1.1 uch wa.accessops = &_gsfb_accessops;
301 1.22 martin wa.accesscookie = sc;
302 1.1 uch
303 1.23 thorpej config_found(self, &wa, wsdisplaydevprint, CFARG_EOL);
304 1.1 uch }
305 1.1 uch
306 1.1 uch /*
307 1.1 uch * console
308 1.1 uch */
309 1.1 uch void
310 1.1 uch gsfbcnprobe(struct consdev *cndev)
311 1.1 uch {
312 1.1 uch
313 1.1 uch cndev->cn_pri = CN_INTERNAL;
314 1.1 uch }
315 1.1 uch
316 1.1 uch void
317 1.1 uch gsfbcninit(struct consdev *cndev)
318 1.1 uch {
319 1.1 uch paddr_t paddr = MIPS_KSEG0_TO_PHYS(gsfb_init_cmd_640x480);
320 1.15 christos u_int32_t *buf = (void *)MIPS_PHYS_TO_KSEG1(paddr);
321 1.15 christos long defattr = ATTR_BG_SET(WS_DEFAULT_BG) | ATTR_FG_SET(WS_DEFAULT_FG);
322 1.1 uch
323 1.22 martin gsfb_is_console = 1;
324 1.1 uch
325 1.1 uch gsfb_hwinit();
326 1.22 martin gsfb_swinit(&gsfb_console_softc);
327 1.1 uch
328 1.15 christos /* Set the screen to the default background color at boot */
329 1.15 christos buf[28] = gsfb_ansi_psmct32[ATTR_BG_GET(defattr)];
330 1.1 uch gsfb_dma_kick(paddr, sizeof gsfb_init_cmd_640x480);
331 1.1 uch #ifdef GSFB_DEBUG_MONITOR
332 1.1 uch {
333 1.1 uch const struct _gsfb_debug_window *win;
334 1.1 uch int i;
335 1.1 uch
336 1.1 uch for (i = 0; i < 3; i++) {
337 1.1 uch win = &_gsfb_debug_window[i];
338 1.1 uch _gsfb_eraserows(0, win->start, win->nrow, win->attr);
339 1.1 uch }
340 1.1 uch }
341 1.1 uch #endif /* GSFB_DEBUG_MONITOR */
342 1.1 uch
343 1.22 martin wsdisplay_cnattach(&_gsfb_std_screen, &gsfb_console_softc, 0, 0,
344 1.22 martin defattr);
345 1.1 uch }
346 1.1 uch
347 1.1 uch void
348 1.19 cegger gsfb_hwinit(void)
349 1.1 uch {
350 1.10 uch /*
351 1.10 uch gs_init(VESA_1A) hang up on SCPH-50000.
352 1.10 uch use bootloader's setting.
353 1.10 uch EN1 | CRTMOD | MMOD | AMOD | ALP(all 1.0)
354 1.10 uch */
355 1.10 uch _reg_write_8(GS_S_PMODE_REG, 0xffa5);
356 1.10 uch
357 1.1 uch dmac_init();
358 1.1 uch
359 1.1 uch /* reset GIF channel DMA */
360 1.1 uch _reg_write_4(D2_QWC_REG, 0);
361 1.1 uch _reg_write_4(D2_MADR_REG, 0);
362 1.1 uch _reg_write_4(D2_TADR_REG, 0);
363 1.1 uch _reg_write_4(D2_CHCR_REG, 0);
364 1.1 uch }
365 1.1 uch
366 1.1 uch int
367 1.22 martin gsfb_swinit(struct gsfb_softc *sc)
368 1.1 uch {
369 1.1 uch int font;
370 1.1 uch
371 1.1 uch wsfont_init();
372 1.22 martin font = wsfont_find(NULL, 8, 16, 0, WSDISPLAY_FONTORDER_L2R,
373 1.22 martin WSDISPLAY_FONTORDER_L2R, WSFONT_FIND_BITMAP);
374 1.1 uch if (font < 0)
375 1.1 uch return (1);
376 1.1 uch
377 1.22 martin if (wsfont_lock(font, &sc->sc_font))
378 1.1 uch return (1);
379 1.1 uch
380 1.22 martin sc->sc_screen = &_gsfb_std_screen;
381 1.1 uch
382 1.1 uch return (0);
383 1.1 uch }
384 1.1 uch
385 1.1 uch /*
386 1.1 uch * wsdisplay
387 1.1 uch */
388 1.1 uch void
389 1.1 uch _gsfb_cursor(void *cookie, int on, int row, int col)
390 1.1 uch {
391 1.22 martin struct gsfb_softc *sc = cookie;
392 1.1 uch paddr_t paddr = MIPS_KSEG0_TO_PHYS(gsfb_cursor_cmd);
393 1.1 uch u_int32_t *buf = (void *)MIPS_PHYS_TO_KSEG1(paddr);
394 1.22 martin struct wsdisplay_font *font = sc->sc_font;
395 1.1 uch
396 1.1 uch gsfb_set_cursor_pos(buf, col, row, font->fontwidth, font->fontheight);
397 1.1 uch
398 1.1 uch gsfb_dma_kick(paddr, sizeof gsfb_cursor_cmd);
399 1.1 uch }
400 1.1 uch
401 1.13 perry inline void
402 1.1 uch gsfb_set_cursor_pos(u_int32_t *p, int x, int y, int w, int h)
403 1.1 uch {
404 1.1 uch
405 1.1 uch x *= w;
406 1.1 uch y *= h;
407 1.1 uch p[20] = ((x << 4) & 0xffff) | ((y << 20) & 0xffff0000);
408 1.14 martin p[28] = (((x + w) << 4) & 0xffff) | (((y + h) << 20) & 0xffff0000);
409 1.1 uch }
410 1.1 uch
411 1.1 uch int
412 1.1 uch _gsfb_mapchar(void *cookie, int c, unsigned int *cp)
413 1.1 uch {
414 1.22 martin struct gsfb_softc *sc = cookie;
415 1.22 martin struct wsdisplay_font *font = sc->sc_font;
416 1.1 uch
417 1.1 uch if (font->encoding != WSDISPLAY_FONTENC_ISO)
418 1.1 uch if ((c = wsfont_map_unichar(font, c)) < 0)
419 1.1 uch goto nomap;
420 1.1 uch
421 1.1 uch if (c < font->firstchar || c >= font->firstchar + font->numchars)
422 1.1 uch goto nomap;
423 1.1 uch
424 1.1 uch *cp = c;
425 1.1 uch return (5);
426 1.1 uch
427 1.1 uch nomap:
428 1.1 uch *cp = ' ';
429 1.1 uch return (0);
430 1.1 uch }
431 1.1 uch
432 1.1 uch void
433 1.1 uch _gsfb_putchar(void *cookie, int row, int col, u_int uc, long attr)
434 1.1 uch {
435 1.22 martin struct gsfb_softc *sc = cookie;
436 1.1 uch paddr_t paddr = MIPS_KSEG0_TO_PHYS(gsfb_load_cmd_8x16_psmct32);
437 1.1 uch u_int32_t *buf = (void *)MIPS_PHYS_TO_KSEG1(paddr);
438 1.22 martin struct wsdisplay_font *font = sc->sc_font;
439 1.1 uch
440 1.1 uch /* copy font data to DMA region */
441 1.1 uch gsfb_font_expand_psmct32(font, uc, attr, &buf[FONT_SCRATCH_BASE]);
442 1.1 uch
443 1.1 uch /* set destination position */
444 1.1 uch TRXPOS_DXY(buf, col * font->fontwidth, row * font->fontheight);
445 1.1 uch
446 1.1 uch /* kick to GIF */
447 1.1 uch gsfb_dma_kick(paddr, sizeof gsfb_load_cmd_8x16_psmct32);
448 1.1 uch }
449 1.1 uch
450 1.1 uch void
451 1.1 uch _gsfb_copycols(void *cookie, int row, int srccol, int dstcol, int ncols)
452 1.1 uch {
453 1.22 martin struct gsfb_softc *sc = cookie;
454 1.1 uch paddr_t paddr = MIPS_KSEG0_TO_PHYS(gsfb_copy_cmd_8x16);
455 1.1 uch u_int32_t *cmd = (void *)MIPS_PHYS_TO_KSEG1(paddr);
456 1.22 martin int y = sc->sc_font->fontheight * row;
457 1.22 martin int w = sc->sc_font->fontwidth;
458 1.1 uch int i;
459 1.1 uch
460 1.1 uch if (dstcol > srccol) {
461 1.1 uch for (i = ncols - 1; i >= 0; i--) {
462 1.1 uch TRXPOS_DXY_SXY(cmd, (dstcol + i) * w, y, (srccol + i) * w, y);
463 1.1 uch gsfb_dma_kick(paddr, sizeof gsfb_copy_cmd_8x16);
464 1.1 uch }
465 1.1 uch } else {
466 1.1 uch for (i = 0; i < ncols; i++) {
467 1.1 uch TRXPOS_DXY_SXY(cmd, (dstcol + i) * w, y, (srccol + i) * w, y);
468 1.1 uch gsfb_dma_kick(paddr, sizeof gsfb_copy_cmd_8x16);
469 1.1 uch }
470 1.1 uch }
471 1.1 uch }
472 1.1 uch
473 1.1 uch void
474 1.1 uch _gsfb_erasecols(void *cookie, int row, int startcol, int ncols, long attr)
475 1.1 uch {
476 1.1 uch int i;
477 1.1 uch
478 1.1 uch for (i = 0; i < ncols; i++)
479 1.1 uch _gsfb_putchar(cookie, row, startcol + i, ' ', attr);
480 1.1 uch }
481 1.1 uch
482 1.1 uch void
483 1.1 uch _gsfb_copyrows(void *cookie, int src, int dst, int num)
484 1.1 uch {
485 1.22 martin struct gsfb_softc *sc = cookie;
486 1.1 uch paddr_t paddr = MIPS_KSEG0_TO_PHYS(gsfb_scroll_cmd_640x16);
487 1.1 uch u_int32_t *cmd = (void *)MIPS_PHYS_TO_KSEG1(paddr);
488 1.1 uch int i;
489 1.22 martin int h = sc->sc_font->fontheight;
490 1.1 uch
491 1.1 uch if (dst > src) {
492 1.1 uch for (i = num - 1; i >= 0; i--) {
493 1.1 uch TRXPOS_SY_DY(cmd, (src + i) * h, (dst + i) * h);
494 1.1 uch gsfb_dma_kick(paddr, sizeof gsfb_scroll_cmd_640x16);
495 1.1 uch }
496 1.1 uch } else {
497 1.1 uch for (i = 0; i < num; i++) {
498 1.1 uch TRXPOS_SY_DY(cmd, (src + i) * h, (dst + i) * h);
499 1.1 uch gsfb_dma_kick(paddr, sizeof gsfb_scroll_cmd_640x16);
500 1.1 uch }
501 1.1 uch }
502 1.1 uch }
503 1.1 uch
504 1.1 uch void
505 1.1 uch _gsfb_eraserows(void *cookie, int row, int nrow, long attr)
506 1.1 uch {
507 1.22 martin struct gsfb_softc *sc = cookie;
508 1.1 uch int i, j;
509 1.1 uch
510 1.1 uch for (j = 0; j < nrow; j++)
511 1.22 martin for (i = 0; i < sc->sc_screen->ncols; i++)
512 1.1 uch _gsfb_putchar(cookie, row + j, i, ' ', attr);
513 1.1 uch }
514 1.1 uch
515 1.1 uch int
516 1.4 junyoung _gsfb_allocattr(void *cookie, int fg, int bg, int flags, long *attr)
517 1.1 uch {
518 1.1 uch
519 1.1 uch if ((flags & WSATTR_BLINK) != 0)
520 1.1 uch return (EINVAL);
521 1.1 uch
522 1.1 uch if ((flags & WSATTR_WSCOLORS) == 0) {
523 1.15 christos fg = WS_DEFAULT_FG;
524 1.15 christos bg = WS_DEFAULT_BG;
525 1.1 uch }
526 1.1 uch
527 1.1 uch if ((flags & WSATTR_HILIT) != 0)
528 1.1 uch fg += 8;
529 1.1 uch
530 1.1 uch flags = (flags & WSATTR_UNDERLINE) ? 1 : 0;
531 1.1 uch
532 1.1 uch
533 1.1 uch *attr = ATTR_BG_SET(bg) | ATTR_FG_SET(fg) | flags;
534 1.1 uch
535 1.1 uch return (0);
536 1.1 uch }
537 1.1 uch
538 1.1 uch int
539 1.17 christos _gsfb_ioctl(void *v, void *vs, u_long cmd, void *data, int flag,
540 1.16 jmmv struct lwp *l)
541 1.1 uch {
542 1.1 uch
543 1.3 atatat return (EPASSTHROUGH); /* Inappropriate ioctl for device */
544 1.1 uch }
545 1.1 uch
546 1.1 uch paddr_t
547 1.16 jmmv _gsfb_mmap(void *v, void *vs, off_t offset, int prot)
548 1.1 uch {
549 1.1 uch
550 1.9 shin return (-1); /* can't mmap */
551 1.1 uch }
552 1.1 uch
553 1.1 uch int
554 1.1 uch _gsfb_alloc_screen(void *v, const struct wsscreen_descr *type, void **cookiep,
555 1.1 uch int *curxp, int *curyp, long *attrp)
556 1.1 uch {
557 1.1 uch
558 1.15 christos *attrp = ATTR_BG_SET(WS_DEFAULT_BG) | ATTR_FG_SET(WS_DEFAULT_FG);
559 1.1 uch
560 1.1 uch return (0);
561 1.1 uch }
562 1.1 uch
563 1.1 uch void
564 1.1 uch _gsfb_free_screen(void *v, void *cookie)
565 1.1 uch {
566 1.1 uch }
567 1.1 uch
568 1.1 uch int
569 1.1 uch _gsfb_show_screen(void *v, void *cookie, int waitok,
570 1.1 uch void (*cb)(void *, int, int), void *cbarg)
571 1.1 uch {
572 1.1 uch
573 1.1 uch return (0);
574 1.1 uch }
575 1.1 uch
576 1.1 uch void
577 1.1 uch _gsfb_pollc(void *v, int on)
578 1.1 uch {
579 1.1 uch
580 1.1 uch }
581 1.1 uch
582 1.1 uch /*
583 1.1 uch * font expansion
584 1.1 uch * PSMCT32 only
585 1.1 uch */
586 1.1 uch void
587 1.1 uch gsfb_font_expand_psmct32(const struct wsdisplay_font *font, u_int c, long attr,
588 1.1 uch u_int32_t *buf)
589 1.1 uch {
590 1.1 uch u_int32_t fg, bg;
591 1.1 uch u_int8_t *bitmap;
592 1.1 uch int i, j;
593 1.1 uch
594 1.1 uch KDASSERT(((u_int32_t)buf & 15) == 0);
595 1.1 uch
596 1.1 uch fg = gsfb_ansi_psmct32[ATTR_FG_GET(attr)];
597 1.1 uch bg = gsfb_ansi_psmct32[ATTR_BG_GET(attr)];
598 1.1 uch
599 1.1 uch bitmap = (u_int8_t *)font->data + (c - font->firstchar) *
600 1.1 uch font->fontheight * font->stride;
601 1.1 uch for (i = 0; i < font->fontheight; i++, bitmap++) {
602 1.1 uch u_int32_t b = *bitmap;
603 1.1 uch for (j = 0; j < font->fontwidth; j++, b <<= 1)
604 1.1 uch *buf++ = (b & 0x80) ? fg : bg;
605 1.1 uch }
606 1.1 uch }
607 1.1 uch
608 1.1 uch void
609 1.1 uch gsfb_dma_kick(paddr_t addr, size_t size)
610 1.1 uch {
611 1.1 uch /* Wait for previous DMA request complete */
612 1.1 uch while (_reg_read_4(D2_QWC_REG))
613 1.1 uch ;
614 1.1 uch
615 1.1 uch /* Wait until GS FIFO empty */
616 1.1 uch while ((_reg_read_8(GS_S_CSR_REG) & (3 << 14)) != (1 << 14))
617 1.1 uch ;
618 1.1 uch
619 1.1 uch /* wait for DMA complete */
620 1.1 uch dmac_bus_poll(D_CH2_GIF);
621 1.1 uch
622 1.1 uch /* transfer addr */
623 1.1 uch _reg_write_4(D2_MADR_REG, addr);
624 1.1 uch /* transfer data size (unit qword) */
625 1.1 uch _reg_write_4(D2_QWC_REG, bytetoqwc(size));
626 1.1 uch
627 1.1 uch /* kick DMA (normal-mode) */
628 1.1 uch dmac_chcr_write(D_CH2_GIF, D_CHCR_STR);
629 1.1 uch }
630 1.1 uch
631 1.1 uch #ifdef GSFB_DEBUG_MONITOR
632 1.1 uch void
633 1.1 uch __gsfb_print(int window, const char *fmt, ...)
634 1.1 uch {
635 1.1 uch const struct _gsfb_debug_window *win;
636 1.1 uch int i, s, x, y, n, a;
637 1.1 uch u_int c;
638 1.1 uch va_list ap;
639 1.1 uch
640 1.1 uch if (!gsfb.initialized)
641 1.1 uch return;
642 1.1 uch
643 1.1 uch s = _intr_suspend();
644 1.1 uch win = &_gsfb_debug_window[window];
645 1.1 uch x = 0;
646 1.1 uch y = win->start;
647 1.1 uch n = win->nrow * 80;
648 1.1 uch a = win->attr;
649 1.1 uch
650 1.1 uch va_start(ap, fmt);
651 1.1 uch vsnprintf(_gsfb_debug_buf, n, fmt, ap);
652 1.1 uch va_end(ap);
653 1.1 uch
654 1.1 uch _gsfb_eraserows(0, y, win->nrow, a);
655 1.1 uch
656 1.1 uch for (i = 0; i < n &&
657 1.1 uch (c = (u_int)_gsfb_debug_buf[i] & 0x7f) != 0; i++) {
658 1.1 uch if (c == '\n')
659 1.1 uch x = 0, y++;
660 1.1 uch else
661 1.1 uch _gsfb_putchar(0, y, x++, c, a);
662 1.1 uch }
663 1.1 uch
664 1.1 uch _intr_resume(s);
665 1.1 uch }
666 1.1 uch
667 1.1 uch void
668 1.1 uch __gsfb_print_hex(int a0, int a1, int a2, int a3)
669 1.1 uch {
670 1.1 uch __gsfb_print(2, "a0=%08x a1=%08x a2=%08x a3=%08x",
671 1.1 uch a0, a1, a2, a3);
672 1.1 uch }
673 1.1 uch #endif /* GSFB_DEBUG_MONITOR */
674