Commit 1fc3d392009d5de7f2787e4e5acea299f6c3ff1e
1 parent
f2c7ba15
g364 framebuffer device
(Hervé Poussineau) git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@4127 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
2 changed files
with
439 additions
and
0 deletions
hw/g364fb.c
0 → 100644
| 1 | +/* | |
| 2 | + * QEMU G364 framebuffer Emulator. | |
| 3 | + * | |
| 4 | + * Copyright (c) 2007-2008 Hervé Poussineau | |
| 5 | + * | |
| 6 | + * This program is free software; you can redistribute it and/or | |
| 7 | + * modify it under the terms of the GNU General Public License as | |
| 8 | + * published by the Free Software Foundation; either version 2 of | |
| 9 | + * the License, or (at your option) any later version. | |
| 10 | + * | |
| 11 | + * This program is distributed in the hope that it will be useful, | |
| 12 | + * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
| 13 | + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
| 14 | + * GNU General Public License for more details. | |
| 15 | + * | |
| 16 | + * You should have received a copy of the GNU General Public License | |
| 17 | + * along with this program; if not, write to the Free Software | |
| 18 | + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, | |
| 19 | + * MA 02111-1307 USA | |
| 20 | + */ | |
| 21 | + | |
| 22 | +#include "hw.h" | |
| 23 | +#include "console.h" | |
| 24 | +#include "pixel_ops.h" | |
| 25 | + | |
| 26 | +//#define DEBUG_G364 | |
| 27 | + | |
| 28 | +typedef struct G364State { | |
| 29 | + target_phys_addr_t vram_base; | |
| 30 | + unsigned int vram_size; | |
| 31 | + uint8_t *vram_buffer; | |
| 32 | + uint32_t ctla; | |
| 33 | + uint8_t palette[256][3]; | |
| 34 | + /* display refresh support */ | |
| 35 | + DisplayState *ds; | |
| 36 | + int graphic_mode; | |
| 37 | + uint32_t scr_width, scr_height; /* in pixels */ | |
| 38 | + uint32_t last_scr_width, last_scr_height; /* in pixels */ | |
| 39 | +} G364State; | |
| 40 | + | |
| 41 | +/* | |
| 42 | + * graphic modes | |
| 43 | + */ | |
| 44 | +#define BPP 8 | |
| 45 | +#define PIXEL_WIDTH 8 | |
| 46 | +#include "g364fb_template.h" | |
| 47 | +#undef BPP | |
| 48 | +#undef PIXEL_WIDTH | |
| 49 | + | |
| 50 | +#define BPP 15 | |
| 51 | +#define PIXEL_WIDTH 16 | |
| 52 | +#include "g364fb_template.h" | |
| 53 | +#undef BPP | |
| 54 | +#undef PIXEL_WIDTH | |
| 55 | + | |
| 56 | +#define BPP 16 | |
| 57 | +#define PIXEL_WIDTH 16 | |
| 58 | +#include "g364fb_template.h" | |
| 59 | +#undef BPP | |
| 60 | +#undef PIXEL_WIDTH | |
| 61 | + | |
| 62 | +#define BPP 32 | |
| 63 | +#define PIXEL_WIDTH 32 | |
| 64 | +#include "g364fb_template.h" | |
| 65 | +#undef BPP | |
| 66 | +#undef PIXEL_WIDTH | |
| 67 | + | |
| 68 | +#define REG_DISPLAYX 0x0918 | |
| 69 | +#define REG_DISPLAYY 0x0940 | |
| 70 | + | |
| 71 | +#define CTLA_FORCE_BLANK 0x400 | |
| 72 | + | |
| 73 | +static void g364fb_draw_graphic(G364State *s, int full_update) | |
| 74 | +{ | |
| 75 | + if (s->scr_width == 0 || s->scr_height == 0) | |
| 76 | + return; | |
| 77 | + if (s->scr_width != s->last_scr_width | |
| 78 | + || s->scr_height != s->last_scr_height) { | |
| 79 | + s->last_scr_width = s->scr_width; | |
| 80 | + s->last_scr_height = s->scr_height; | |
| 81 | + dpy_resize(s->ds, s->last_scr_width, s->last_scr_height); | |
| 82 | + full_update = 1; | |
| 83 | + } | |
| 84 | + | |
| 85 | + switch (s->ds->depth) { | |
| 86 | + case 8: | |
| 87 | + g364fb_draw_graphic8(s, full_update); | |
| 88 | + break; | |
| 89 | + case 15: | |
| 90 | + g364fb_draw_graphic15(s, full_update); | |
| 91 | + break; | |
| 92 | + case 16: | |
| 93 | + g364fb_draw_graphic16(s, full_update); | |
| 94 | + break; | |
| 95 | + case 32: | |
| 96 | + g364fb_draw_graphic32(s, full_update); | |
| 97 | + break; | |
| 98 | + default: | |
| 99 | + printf("g364fb: unknown depth %d\n", s->ds->depth); | |
| 100 | + return; | |
| 101 | + } | |
| 102 | + | |
| 103 | + dpy_update(s->ds, 0, 0, s->last_scr_width, s->last_scr_height); | |
| 104 | +} | |
| 105 | + | |
| 106 | +static void g364fb_draw_blank(G364State *s, int full_update) | |
| 107 | +{ | |
| 108 | + int i, w; | |
| 109 | + uint8_t *d; | |
| 110 | + | |
| 111 | + if (!full_update) | |
| 112 | + return; | |
| 113 | + if (s->last_scr_width <= 0 || s->last_scr_height <= 0) | |
| 114 | + return; | |
| 115 | + | |
| 116 | + w = s->last_scr_width * ((s->ds->depth + 7) >> 3); | |
| 117 | + d = s->ds->data; | |
| 118 | + for(i = 0; i < s->last_scr_height; i++) { | |
| 119 | + memset(d, 0, w); | |
| 120 | + d += s->ds->linesize; | |
| 121 | + } | |
| 122 | + dpy_update(s->ds, 0, 0, | |
| 123 | + s->last_scr_width, s->last_scr_height); | |
| 124 | +} | |
| 125 | + | |
| 126 | +#define GMODE_GRAPH 0 | |
| 127 | +#define GMODE_BLANK 1 | |
| 128 | + | |
| 129 | +static void g364fb_update_display(void *opaque) | |
| 130 | +{ | |
| 131 | + G364State *s = opaque; | |
| 132 | + int full_update, graphic_mode; | |
| 133 | + | |
| 134 | + if (s->ctla & CTLA_FORCE_BLANK) | |
| 135 | + graphic_mode = GMODE_BLANK; | |
| 136 | + else | |
| 137 | + graphic_mode = GMODE_GRAPH; | |
| 138 | + full_update = 0; | |
| 139 | + if (graphic_mode != s->graphic_mode) { | |
| 140 | + s->graphic_mode = graphic_mode; | |
| 141 | + full_update = 1; | |
| 142 | + } | |
| 143 | + switch(graphic_mode) { | |
| 144 | + case GMODE_GRAPH: | |
| 145 | + g364fb_draw_graphic(s, full_update); | |
| 146 | + break; | |
| 147 | + case GMODE_BLANK: | |
| 148 | + default: | |
| 149 | + g364fb_draw_blank(s, full_update); | |
| 150 | + break; | |
| 151 | + } | |
| 152 | +} | |
| 153 | + | |
| 154 | +/* force a full display refresh */ | |
| 155 | +static void g364fb_invalidate_display(void *opaque) | |
| 156 | +{ | |
| 157 | + G364State *s = opaque; | |
| 158 | + s->graphic_mode = -1; /* force full update */ | |
| 159 | +} | |
| 160 | + | |
| 161 | +static void g364fb_reset(void *opaque) | |
| 162 | +{ | |
| 163 | + G364State *s = opaque; | |
| 164 | + | |
| 165 | + memset(s->palette, 0, sizeof(s->palette)); | |
| 166 | + s->scr_width = s->scr_height = 0; | |
| 167 | + s->last_scr_width = s->last_scr_height = 0; | |
| 168 | + memset(s->vram_buffer, 0, s->vram_size); | |
| 169 | + s->graphic_mode = -1; /* force full update */ | |
| 170 | +} | |
| 171 | + | |
| 172 | +static void g364fb_screen_dump(void *opaque, const char *filename) | |
| 173 | +{ | |
| 174 | + G364State *s = opaque; | |
| 175 | + int y, x; | |
| 176 | + uint8_t index; | |
| 177 | + uint8_t *data_buffer; | |
| 178 | + FILE *f; | |
| 179 | + | |
| 180 | + f = fopen(filename, "wb"); | |
| 181 | + if (!f) | |
| 182 | + return; | |
| 183 | + | |
| 184 | + data_buffer = s->vram_buffer; | |
| 185 | + fprintf(f, "P6\n%d %d\n%d\n", | |
| 186 | + s->scr_width, s->scr_height, 255); | |
| 187 | + for(y = 0; y < s->scr_height; y++) | |
| 188 | + for(x = 0; x < s->scr_width; x++, data_buffer++) { | |
| 189 | + index = *data_buffer; | |
| 190 | + fputc(s->palette[index][0], f); | |
| 191 | + fputc(s->palette[index][1], f); | |
| 192 | + fputc(s->palette[index][2], f); | |
| 193 | + } | |
| 194 | + fclose(f); | |
| 195 | +} | |
| 196 | + | |
| 197 | +/* called for accesses to io ports */ | |
| 198 | +static uint32_t g364fb_ctrl_readb(void *opaque, target_phys_addr_t addr) | |
| 199 | +{ | |
| 200 | + //G364State *s = opaque; | |
| 201 | + uint32_t val; | |
| 202 | + | |
| 203 | + addr &= 0xffff; | |
| 204 | + | |
| 205 | + switch (addr) { | |
| 206 | + default: | |
| 207 | +#ifdef DEBUG_G364 | |
| 208 | + printf("g364fb/ctrl: invalid read at [" TARGET_FMT_lx "]\n", addr); | |
| 209 | +#endif | |
| 210 | + val = 0; | |
| 211 | + break; | |
| 212 | + } | |
| 213 | + | |
| 214 | +#ifdef DEBUG_G364 | |
| 215 | + printf("g364fb/ctrl: read 0x%02x at [" TARGET_FMT_lx "]\n", val, addr); | |
| 216 | +#endif | |
| 217 | + | |
| 218 | + return val; | |
| 219 | +} | |
| 220 | + | |
| 221 | +static uint32_t g364fb_ctrl_readw(void *opaque, target_phys_addr_t addr) | |
| 222 | +{ | |
| 223 | + uint32_t v; | |
| 224 | + v = g364fb_ctrl_readb(opaque, addr); | |
| 225 | + v |= g364fb_ctrl_readb(opaque, addr + 1) << 8; | |
| 226 | + return v; | |
| 227 | +} | |
| 228 | + | |
| 229 | +static uint32_t g364fb_ctrl_readl(void *opaque, target_phys_addr_t addr) | |
| 230 | +{ | |
| 231 | + uint32_t v; | |
| 232 | + v = g364fb_ctrl_readb(opaque, addr); | |
| 233 | + v |= g364fb_ctrl_readb(opaque, addr + 1) << 8; | |
| 234 | + v |= g364fb_ctrl_readb(opaque, addr + 2) << 16; | |
| 235 | + v |= g364fb_ctrl_readb(opaque, addr + 3) << 24; | |
| 236 | + return v; | |
| 237 | +} | |
| 238 | + | |
| 239 | +static void g364fb_ctrl_writeb(void *opaque, target_phys_addr_t addr, uint32_t val) | |
| 240 | +{ | |
| 241 | + G364State *s = opaque; | |
| 242 | + | |
| 243 | + addr &= 0xffff; | |
| 244 | + | |
| 245 | +#ifdef DEBUG_G364 | |
| 246 | + printf("g364fb/ctrl: write 0x%02x at [" TARGET_FMT_lx "]\n", val, addr); | |
| 247 | +#endif | |
| 248 | + | |
| 249 | + if (addr < 0x0800) { | |
| 250 | + /* color palette */ | |
| 251 | + int idx = addr >> 3; | |
| 252 | + int c = addr & 7; | |
| 253 | + if (c < 3) | |
| 254 | + s->palette[idx][c] = (uint8_t)val; | |
| 255 | + } else { | |
| 256 | + switch (addr) { | |
| 257 | + case REG_DISPLAYX: | |
| 258 | + s->scr_width = (s->scr_width & 0xfffffc03) | (val << 2); | |
| 259 | + break; | |
| 260 | + case REG_DISPLAYX + 1: | |
| 261 | + s->scr_width = (s->scr_width & 0xfffc03ff) | (val << 10); | |
| 262 | + break; | |
| 263 | + case REG_DISPLAYY: | |
| 264 | + s->scr_height = (s->scr_height & 0xffffff80) | (val >> 1); | |
| 265 | + break; | |
| 266 | + case REG_DISPLAYY + 1: | |
| 267 | + s->scr_height = (s->scr_height & 0xffff801f) | (val << 7); | |
| 268 | + break; | |
| 269 | + default: | |
| 270 | +#ifdef DEBUG_G364 | |
| 271 | + printf("g364fb/ctrl: invalid write of 0x%02x at [" TARGET_FMT_lx "]\n", val, addr); | |
| 272 | +#endif | |
| 273 | + break; | |
| 274 | + } | |
| 275 | + } | |
| 276 | +} | |
| 277 | + | |
| 278 | +static void g364fb_ctrl_writew(void *opaque, target_phys_addr_t addr, uint32_t val) | |
| 279 | +{ | |
| 280 | + g364fb_ctrl_writeb(opaque, addr, val & 0xff); | |
| 281 | + g364fb_ctrl_writeb(opaque, addr + 1, (val >> 8) & 0xff); | |
| 282 | +} | |
| 283 | + | |
| 284 | +static void g364fb_ctrl_writel(void *opaque, target_phys_addr_t addr, uint32_t val) | |
| 285 | +{ | |
| 286 | + g364fb_ctrl_writeb(opaque, addr, val & 0xff); | |
| 287 | + g364fb_ctrl_writeb(opaque, addr + 1, (val >> 8) & 0xff); | |
| 288 | + g364fb_ctrl_writeb(opaque, addr + 2, (val >> 16) & 0xff); | |
| 289 | + g364fb_ctrl_writeb(opaque, addr + 3, (val >> 24) & 0xff); | |
| 290 | +} | |
| 291 | + | |
| 292 | +static CPUReadMemoryFunc *g364fb_ctrl_read[3] = { | |
| 293 | + g364fb_ctrl_readb, | |
| 294 | + g364fb_ctrl_readw, | |
| 295 | + g364fb_ctrl_readl, | |
| 296 | +}; | |
| 297 | + | |
| 298 | +static CPUWriteMemoryFunc *g364fb_ctrl_write[3] = { | |
| 299 | + g364fb_ctrl_writeb, | |
| 300 | + g364fb_ctrl_writew, | |
| 301 | + g364fb_ctrl_writel, | |
| 302 | +}; | |
| 303 | + | |
| 304 | +/* called for accesses to video ram */ | |
| 305 | +static uint32_t g364fb_mem_readb(void *opaque, target_phys_addr_t addr) | |
| 306 | +{ | |
| 307 | + G364State *s = opaque; | |
| 308 | + target_phys_addr_t relative_addr = addr - s->vram_base; | |
| 309 | + | |
| 310 | + return s->vram_buffer[relative_addr]; | |
| 311 | +} | |
| 312 | + | |
| 313 | +static uint32_t g364fb_mem_readw(void *opaque, target_phys_addr_t addr) | |
| 314 | +{ | |
| 315 | + uint32_t v; | |
| 316 | + v = g364fb_mem_readb(opaque, addr); | |
| 317 | + v |= g364fb_mem_readb(opaque, addr + 1) << 8; | |
| 318 | + return v; | |
| 319 | +} | |
| 320 | + | |
| 321 | +static uint32_t g364fb_mem_readl(void *opaque, target_phys_addr_t addr) | |
| 322 | +{ | |
| 323 | + uint32_t v; | |
| 324 | + v = g364fb_mem_readb(opaque, addr); | |
| 325 | + v |= g364fb_mem_readb(opaque, addr + 1) << 8; | |
| 326 | + v |= g364fb_mem_readb(opaque, addr + 2) << 16; | |
| 327 | + v |= g364fb_mem_readb(opaque, addr + 3) << 24; | |
| 328 | + return v; | |
| 329 | +} | |
| 330 | + | |
| 331 | +static void g364fb_mem_writeb(void *opaque, target_phys_addr_t addr, uint32_t val) | |
| 332 | +{ | |
| 333 | + G364State *s = opaque; | |
| 334 | + target_phys_addr_t relative_addr = addr - s->vram_base; | |
| 335 | + | |
| 336 | + s->vram_buffer[relative_addr] = val; | |
| 337 | +} | |
| 338 | + | |
| 339 | +static void g364fb_mem_writew(void *opaque, target_phys_addr_t addr, uint32_t val) | |
| 340 | +{ | |
| 341 | + g364fb_mem_writeb(opaque, addr, val & 0xff); | |
| 342 | + g364fb_mem_writeb(opaque, addr + 1, (val >> 8) & 0xff); | |
| 343 | +} | |
| 344 | + | |
| 345 | +static void g364fb_mem_writel(void *opaque, target_phys_addr_t addr, uint32_t val) | |
| 346 | +{ | |
| 347 | + g364fb_mem_writeb(opaque, addr, val & 0xff); | |
| 348 | + g364fb_mem_writeb(opaque, addr + 1, (val >> 8) & 0xff); | |
| 349 | + g364fb_mem_writeb(opaque, addr + 2, (val >> 16) & 0xff); | |
| 350 | + g364fb_mem_writeb(opaque, addr + 3, (val >> 24) & 0xff); | |
| 351 | +} | |
| 352 | + | |
| 353 | +static CPUReadMemoryFunc *g364fb_mem_read[3] = { | |
| 354 | + g364fb_mem_readb, | |
| 355 | + g364fb_mem_readw, | |
| 356 | + g364fb_mem_readl, | |
| 357 | +}; | |
| 358 | + | |
| 359 | +static CPUWriteMemoryFunc *g364fb_mem_write[3] = { | |
| 360 | + g364fb_mem_writeb, | |
| 361 | + g364fb_mem_writew, | |
| 362 | + g364fb_mem_writel, | |
| 363 | +}; | |
| 364 | + | |
| 365 | +int g364fb_mm_init(DisplayState *ds, | |
| 366 | + int vram_size, int it_shift, | |
| 367 | + target_phys_addr_t vram_base, target_phys_addr_t ctrl_base) | |
| 368 | +{ | |
| 369 | + G364State *s; | |
| 370 | + int io_vram, io_ctrl; | |
| 371 | + | |
| 372 | + s = qemu_mallocz(sizeof(G364State)); | |
| 373 | + if (!s) | |
| 374 | + return -1; | |
| 375 | + | |
| 376 | + s->vram_size = vram_size; | |
| 377 | + s->vram_buffer = qemu_mallocz(s->vram_size); | |
| 378 | + | |
| 379 | + qemu_register_reset(g364fb_reset, s); | |
| 380 | + g364fb_reset(s); | |
| 381 | + | |
| 382 | + s->ds = ds; | |
| 383 | + s->vram_base = vram_base; | |
| 384 | + | |
| 385 | + graphic_console_init(ds, g364fb_update_display, | |
| 386 | + g364fb_invalidate_display, g364fb_screen_dump, | |
| 387 | + NULL, s); | |
| 388 | + | |
| 389 | + io_vram = cpu_register_io_memory(0, g364fb_mem_read, g364fb_mem_write, s); | |
| 390 | + cpu_register_physical_memory(s->vram_base, vram_size, io_vram); | |
| 391 | + | |
| 392 | + io_ctrl = cpu_register_io_memory(0, g364fb_ctrl_read, g364fb_ctrl_write, s); | |
| 393 | + cpu_register_physical_memory(ctrl_base, 0x10000, io_ctrl); | |
| 394 | + | |
| 395 | + return 0; | |
| 396 | +} | ... | ... |
hw/g364fb_template.h
0 → 100644
| 1 | +/* | |
| 2 | + * QEMU G364 framebuffer Emulator. | |
| 3 | + * | |
| 4 | + * Copyright (c) 2007 Hervé Poussineau | |
| 5 | + * | |
| 6 | + * This program is free software; you can redistribute it and/or | |
| 7 | + * modify it under the terms of the GNU General Public License as | |
| 8 | + * published by the Free Software Foundation; either version 2 of | |
| 9 | + * the License, or (at your option) any later version. | |
| 10 | + * | |
| 11 | + * This program is distributed in the hope that it will be useful, | |
| 12 | + * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
| 13 | + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
| 14 | + * GNU General Public License for more details. | |
| 15 | + * | |
| 16 | + * You should have received a copy of the GNU General Public License | |
| 17 | + * along with this program; if not, write to the Free Software | |
| 18 | + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, | |
| 19 | + * MA 02111-1307 USA | |
| 20 | + */ | |
| 21 | + | |
| 22 | +static void glue(g364fb_draw_graphic, BPP)(G364State *s, int full_update) | |
| 23 | +{ | |
| 24 | + int i, j; | |
| 25 | + int w_display; | |
| 26 | + uint8_t *data_buffer; | |
| 27 | + uint8_t *data_display, *dd; | |
| 28 | + | |
| 29 | + data_buffer = s->vram_buffer; | |
| 30 | + w_display = s->last_scr_width * PIXEL_WIDTH / 8; | |
| 31 | + data_display = s->ds->data; | |
| 32 | + for(i = 0; i < s->last_scr_height; i++) { | |
| 33 | + dd = data_display; | |
| 34 | + for (j = 0; j < s->last_scr_width; j++, dd += PIXEL_WIDTH / 8, data_buffer++) { | |
| 35 | + uint8_t index = *data_buffer; | |
| 36 | + *((glue(glue(uint, PIXEL_WIDTH), _t) *)dd) = glue(rgb_to_pixel, BPP)( | |
| 37 | + s->palette[index][0], | |
| 38 | + s->palette[index][1], | |
| 39 | + s->palette[index][2]); | |
| 40 | + } | |
| 41 | + data_display += s->ds->linesize; | |
| 42 | + } | |
| 43 | +} | ... | ... |