Commit 7e7c5e4c1ba5c9b7efcf1b0c1e34ea150c286e58
1 parent
a5d7eb65
Nokia N800 machine support (ARM).
Also add various peripherals: two miscellaneous Nokia CBUS chips, EPSON S1D13745 LCD/TV remote-framebuffer controller, TWL92230 - standard OMAP2 power management companion chip on i2c. Generic OneNAND flash memory, TMP105 temperature sensor on i2c. git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@4215 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
15 changed files
with
4540 additions
and
3 deletions
Makefile
| ... | ... | @@ -51,7 +51,8 @@ OBJS+=block.o |
| 51 | 51 | |
| 52 | 52 | OBJS+=irq.o |
| 53 | 53 | OBJS+=i2c.o smbus.o smbus_eeprom.o max7310.o max111x.o wm8750.o |
| 54 | -OBJS+=ssd0303.o ssd0323.o ads7846.o stellaris_input.o | |
| 54 | +OBJS+=ssd0303.o ssd0323.o ads7846.o stellaris_input.o twl92230.o | |
| 55 | +OBJS+=tmp105.o | |
| 55 | 56 | OBJS+=scsi-disk.o cdrom.o |
| 56 | 57 | OBJS+=scsi-generic.o |
| 57 | 58 | OBJS+=usb.o usb-hub.o usb-linux.o usb-hid.o usb-msd.o usb-wacom.o usb-serial.o | ... | ... |
Makefile.target
| ... | ... | @@ -612,6 +612,7 @@ OBJS+= spitz.o ide.o serial.o nand.o ecc.o |
| 612 | 612 | OBJS+= omap1.o omap_lcdc.o omap_dma.o omap_clk.o omap_mmc.o omap_i2c.o |
| 613 | 613 | OBJS+= omap2.o omap_dss.o |
| 614 | 614 | OBJS+= palm.o tsc210x.o |
| 615 | +OBJS+= nseries.o blizzard.o onenand.o vga.o cbus.o | |
| 615 | 616 | OBJS+= mst_fpga.o mainstone.o |
| 616 | 617 | CPPFLAGS += -DHAS_AUDIO |
| 617 | 618 | endif | ... | ... |
hw/blizzard.c
0 → 100644
| 1 | +/* | |
| 2 | + * Epson S1D13744/S1D13745 (Blizzard/Hailstorm/Tornado) LCD/TV controller. | |
| 3 | + * | |
| 4 | + * Copyright (C) 2008 Nokia Corporation | |
| 5 | + * Written by Andrzej Zaborowski <andrew@openedhand.com> | |
| 6 | + * | |
| 7 | + * This program is free software; you can redistribute it and/or | |
| 8 | + * modify it under the terms of the GNU General Public License as | |
| 9 | + * published by the Free Software Foundation; either version 2 or | |
| 10 | + * (at your option) version 3 of the License. | |
| 11 | + * | |
| 12 | + * This program is distributed in the hope that it will be useful, | |
| 13 | + * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
| 14 | + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
| 15 | + * GNU General Public License for more details. | |
| 16 | + * | |
| 17 | + * You should have received a copy of the GNU General Public License | |
| 18 | + * along with this program; if not, write to the Free Software | |
| 19 | + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, | |
| 20 | + * MA 02111-1307 USA | |
| 21 | + */ | |
| 22 | + | |
| 23 | +#include "qemu-common.h" | |
| 24 | +#include "sysemu.h" | |
| 25 | +#include "console.h" | |
| 26 | +#include "devices.h" | |
| 27 | +#include "vga_int.h" | |
| 28 | +#include "pixel_ops.h" | |
| 29 | + | |
| 30 | +typedef void (*blizzard_fn_t)(uint8_t *, const uint8_t *, unsigned int); | |
| 31 | + | |
| 32 | +struct blizzard_s { | |
| 33 | + uint8_t reg; | |
| 34 | + uint32_t addr; | |
| 35 | + int swallow; | |
| 36 | + | |
| 37 | + int pll; | |
| 38 | + int pll_range; | |
| 39 | + int pll_ctrl; | |
| 40 | + uint8_t pll_mode; | |
| 41 | + uint8_t clksel; | |
| 42 | + int memenable; | |
| 43 | + int memrefresh; | |
| 44 | + uint8_t timing[3]; | |
| 45 | + int priority; | |
| 46 | + | |
| 47 | + uint8_t lcd_config; | |
| 48 | + int x; | |
| 49 | + int y; | |
| 50 | + int skipx; | |
| 51 | + int skipy; | |
| 52 | + uint8_t hndp; | |
| 53 | + uint8_t vndp; | |
| 54 | + uint8_t hsync; | |
| 55 | + uint8_t vsync; | |
| 56 | + uint8_t pclk; | |
| 57 | + uint8_t u; | |
| 58 | + uint8_t v; | |
| 59 | + uint8_t yrc[2]; | |
| 60 | + int ix[2]; | |
| 61 | + int iy[2]; | |
| 62 | + int ox[2]; | |
| 63 | + int oy[2]; | |
| 64 | + | |
| 65 | + int enable; | |
| 66 | + int blank; | |
| 67 | + int bpp; | |
| 68 | + int invalidate; | |
| 69 | + int mx[2]; | |
| 70 | + int my[2]; | |
| 71 | + uint8_t mode; | |
| 72 | + uint8_t effect; | |
| 73 | + uint8_t iformat; | |
| 74 | + uint8_t source; | |
| 75 | + DisplayState *state; | |
| 76 | + blizzard_fn_t *line_fn_tab[2]; | |
| 77 | + void *fb; | |
| 78 | + | |
| 79 | + uint8_t hssi_config[3]; | |
| 80 | + uint8_t tv_config; | |
| 81 | + uint8_t tv_timing[4]; | |
| 82 | + uint8_t vbi; | |
| 83 | + uint8_t tv_x; | |
| 84 | + uint8_t tv_y; | |
| 85 | + uint8_t tv_test; | |
| 86 | + uint8_t tv_filter_config; | |
| 87 | + uint8_t tv_filter_idx; | |
| 88 | + uint8_t tv_filter_coeff[0x20]; | |
| 89 | + uint8_t border_r; | |
| 90 | + uint8_t border_g; | |
| 91 | + uint8_t border_b; | |
| 92 | + uint8_t gamma_config; | |
| 93 | + uint8_t gamma_idx; | |
| 94 | + uint8_t gamma_lut[0x100]; | |
| 95 | + uint8_t matrix_ena; | |
| 96 | + uint8_t matrix_coeff[0x12]; | |
| 97 | + uint8_t matrix_r; | |
| 98 | + uint8_t matrix_g; | |
| 99 | + uint8_t matrix_b; | |
| 100 | + uint8_t pm; | |
| 101 | + uint8_t status; | |
| 102 | + uint8_t rgbgpio_dir; | |
| 103 | + uint8_t rgbgpio; | |
| 104 | + uint8_t gpio_dir; | |
| 105 | + uint8_t gpio; | |
| 106 | + uint8_t gpio_edge[2]; | |
| 107 | + uint8_t gpio_irq; | |
| 108 | + uint8_t gpio_pdown; | |
| 109 | + | |
| 110 | + struct { | |
| 111 | + int x; | |
| 112 | + int y; | |
| 113 | + int dx; | |
| 114 | + int dy; | |
| 115 | + int len; | |
| 116 | + int buflen; | |
| 117 | + void *buf; | |
| 118 | + void *data; | |
| 119 | + uint16_t *ptr; | |
| 120 | + int angle; | |
| 121 | + int pitch; | |
| 122 | + blizzard_fn_t line_fn; | |
| 123 | + } data; | |
| 124 | +}; | |
| 125 | + | |
| 126 | +/* Bytes(!) per pixel */ | |
| 127 | +static const int blizzard_iformat_bpp[0x10] = { | |
| 128 | + 0, | |
| 129 | + 2, /* RGB 5:6:5*/ | |
| 130 | + 3, /* RGB 6:6:6 mode 1 */ | |
| 131 | + 3, /* RGB 8:8:8 mode 1 */ | |
| 132 | + 0, 0, | |
| 133 | + 4, /* RGB 6:6:6 mode 2 */ | |
| 134 | + 4, /* RGB 8:8:8 mode 2 */ | |
| 135 | + 0, /* YUV 4:2:2 */ | |
| 136 | + 0, /* YUV 4:2:0 */ | |
| 137 | + 0, 0, 0, 0, 0, 0, | |
| 138 | +}; | |
| 139 | + | |
| 140 | +static inline void blizzard_rgb2yuv(int r, int g, int b, | |
| 141 | + int *y, int *u, int *v) | |
| 142 | +{ | |
| 143 | + *y = 0x10 + ((0x838 * r + 0x1022 * g + 0x322 * b) >> 13); | |
| 144 | + *u = 0x80 + ((0xe0e * b - 0x04c1 * r - 0x94e * g) >> 13); | |
| 145 | + *v = 0x80 + ((0xe0e * r - 0x0bc7 * g - 0x247 * b) >> 13); | |
| 146 | +} | |
| 147 | + | |
| 148 | +static void blizzard_window(struct blizzard_s *s) | |
| 149 | +{ | |
| 150 | + uint8_t *src, *dst; | |
| 151 | + int bypp[2]; | |
| 152 | + int bypl[3]; | |
| 153 | + int y; | |
| 154 | + blizzard_fn_t fn = s->data.line_fn; | |
| 155 | + | |
| 156 | + if (!fn) | |
| 157 | + return; | |
| 158 | + if (s->mx[0] > s->data.x) | |
| 159 | + s->mx[0] = s->data.x; | |
| 160 | + if (s->my[0] > s->data.y) | |
| 161 | + s->my[0] = s->data.y; | |
| 162 | + if (s->mx[1] < s->data.x + s->data.dx) | |
| 163 | + s->mx[1] = s->data.x + s->data.dx; | |
| 164 | + if (s->my[1] < s->data.y + s->data.dy) | |
| 165 | + s->my[1] = s->data.y + s->data.dy; | |
| 166 | + | |
| 167 | + bypp[0] = s->bpp; | |
| 168 | + bypp[1] = (s->state->depth + 7) >> 3; | |
| 169 | + bypl[0] = bypp[0] * s->data.pitch; | |
| 170 | + bypl[1] = bypp[1] * s->x; | |
| 171 | + bypl[2] = bypp[0] * s->data.dx; | |
| 172 | + | |
| 173 | + src = s->data.data; | |
| 174 | + dst = s->fb + bypl[1] * s->data.y + bypp[1] * s->data.x; | |
| 175 | + for (y = s->data.dy; y > 0; y --, src += bypl[0], dst += bypl[1]) | |
| 176 | + fn(dst, src, bypl[2]); | |
| 177 | +} | |
| 178 | + | |
| 179 | +static int blizzard_transfer_setup(struct blizzard_s *s) | |
| 180 | +{ | |
| 181 | + if (s->source > 3 || !s->bpp || | |
| 182 | + s->ix[1] < s->ix[0] || s->iy[1] < s->iy[0]) | |
| 183 | + return 0; | |
| 184 | + | |
| 185 | + s->data.angle = s->effect & 3; | |
| 186 | + s->data.line_fn = s->line_fn_tab[!!s->data.angle][s->iformat]; | |
| 187 | + s->data.x = s->ix[0]; | |
| 188 | + s->data.y = s->iy[0]; | |
| 189 | + s->data.dx = s->ix[1] - s->ix[0] + 1; | |
| 190 | + s->data.dy = s->iy[1] - s->iy[0] + 1; | |
| 191 | + s->data.len = s->bpp * s->data.dx * s->data.dy; | |
| 192 | + s->data.pitch = s->data.dx; | |
| 193 | + if (s->data.len > s->data.buflen) { | |
| 194 | + s->data.buf = realloc(s->data.buf, s->data.len); | |
| 195 | + s->data.buflen = s->data.len; | |
| 196 | + } | |
| 197 | + s->data.ptr = s->data.buf; | |
| 198 | + s->data.data = s->data.buf; | |
| 199 | + s->data.len /= 2; | |
| 200 | + return 1; | |
| 201 | +} | |
| 202 | + | |
| 203 | +static void blizzard_reset(struct blizzard_s *s) | |
| 204 | +{ | |
| 205 | + s->reg = 0; | |
| 206 | + s->swallow = 0; | |
| 207 | + | |
| 208 | + s->pll = 9; | |
| 209 | + s->pll_range = 1; | |
| 210 | + s->pll_ctrl = 0x14; | |
| 211 | + s->pll_mode = 0x32; | |
| 212 | + s->clksel = 0x00; | |
| 213 | + s->memenable = 0; | |
| 214 | + s->memrefresh = 0x25c; | |
| 215 | + s->timing[0] = 0x3f; | |
| 216 | + s->timing[1] = 0x13; | |
| 217 | + s->timing[2] = 0x21; | |
| 218 | + s->priority = 0; | |
| 219 | + | |
| 220 | + s->lcd_config = 0x74; | |
| 221 | + s->x = 8; | |
| 222 | + s->y = 1; | |
| 223 | + s->skipx = 0; | |
| 224 | + s->skipy = 0; | |
| 225 | + s->hndp = 3; | |
| 226 | + s->vndp = 2; | |
| 227 | + s->hsync = 1; | |
| 228 | + s->vsync = 1; | |
| 229 | + s->pclk = 0x80; | |
| 230 | + | |
| 231 | + s->ix[0] = 0; | |
| 232 | + s->ix[1] = 0; | |
| 233 | + s->iy[0] = 0; | |
| 234 | + s->iy[1] = 0; | |
| 235 | + s->ox[0] = 0; | |
| 236 | + s->ox[1] = 0; | |
| 237 | + s->oy[0] = 0; | |
| 238 | + s->oy[1] = 0; | |
| 239 | + | |
| 240 | + s->yrc[0] = 0x00; | |
| 241 | + s->yrc[1] = 0x30; | |
| 242 | + s->u = 0; | |
| 243 | + s->v = 0; | |
| 244 | + | |
| 245 | + s->iformat = 3; | |
| 246 | + s->source = 0; | |
| 247 | + s->bpp = blizzard_iformat_bpp[s->iformat]; | |
| 248 | + | |
| 249 | + s->hssi_config[0] = 0x00; | |
| 250 | + s->hssi_config[1] = 0x00; | |
| 251 | + s->hssi_config[2] = 0x01; | |
| 252 | + s->tv_config = 0x00; | |
| 253 | + s->tv_timing[0] = 0x00; | |
| 254 | + s->tv_timing[1] = 0x00; | |
| 255 | + s->tv_timing[2] = 0x00; | |
| 256 | + s->tv_timing[3] = 0x00; | |
| 257 | + s->vbi = 0x10; | |
| 258 | + s->tv_x = 0x14; | |
| 259 | + s->tv_y = 0x03; | |
| 260 | + s->tv_test = 0x00; | |
| 261 | + s->tv_filter_config = 0x80; | |
| 262 | + s->tv_filter_idx = 0x00; | |
| 263 | + s->border_r = 0x10; | |
| 264 | + s->border_g = 0x80; | |
| 265 | + s->border_b = 0x80; | |
| 266 | + s->gamma_config = 0x00; | |
| 267 | + s->gamma_idx = 0x00; | |
| 268 | + s->matrix_ena = 0x00; | |
| 269 | + memset(&s->matrix_coeff, 0, sizeof(s->matrix_coeff)); | |
| 270 | + s->matrix_r = 0x00; | |
| 271 | + s->matrix_g = 0x00; | |
| 272 | + s->matrix_b = 0x00; | |
| 273 | + s->pm = 0x02; | |
| 274 | + s->status = 0x00; | |
| 275 | + s->rgbgpio_dir = 0x00; | |
| 276 | + s->gpio_dir = 0x00; | |
| 277 | + s->gpio_edge[0] = 0x00; | |
| 278 | + s->gpio_edge[1] = 0x00; | |
| 279 | + s->gpio_irq = 0x00; | |
| 280 | + s->gpio_pdown = 0xff; | |
| 281 | +} | |
| 282 | + | |
| 283 | +static inline void blizzard_invalidate_display(void *opaque) { | |
| 284 | + struct blizzard_s *s = (struct blizzard_s *) opaque; | |
| 285 | + | |
| 286 | + s->invalidate = 1; | |
| 287 | +} | |
| 288 | + | |
| 289 | +static uint16_t blizzard_reg_read(void *opaque, uint8_t reg) | |
| 290 | +{ | |
| 291 | + struct blizzard_s *s = (struct blizzard_s *) opaque; | |
| 292 | + | |
| 293 | + switch (reg) { | |
| 294 | + case 0x00: /* Revision Code */ | |
| 295 | + return 0xa5; | |
| 296 | + | |
| 297 | + case 0x02: /* Configuration Readback */ | |
| 298 | + return 0x83; /* Macrovision OK, CNF[2:0] = 3 */ | |
| 299 | + | |
| 300 | + case 0x04: /* PLL M-Divider */ | |
| 301 | + return (s->pll - 1) | (1 << 7); | |
| 302 | + case 0x06: /* PLL Lock Range Control */ | |
| 303 | + return s->pll_range; | |
| 304 | + case 0x08: /* PLL Lock Synthesis Control 0 */ | |
| 305 | + return s->pll_ctrl & 0xff; | |
| 306 | + case 0x0a: /* PLL Lock Synthesis Control 1 */ | |
| 307 | + return s->pll_ctrl >> 8; | |
| 308 | + case 0x0c: /* PLL Mode Control 0 */ | |
| 309 | + return s->pll_mode; | |
| 310 | + | |
| 311 | + case 0x0e: /* Clock-Source Select */ | |
| 312 | + return s->clksel; | |
| 313 | + | |
| 314 | + case 0x10: /* Memory Controller Activate */ | |
| 315 | + case 0x14: /* Memory Controller Bank 0 Status Flag */ | |
| 316 | + return s->memenable; | |
| 317 | + | |
| 318 | + case 0x18: /* Auto-Refresh Interval Setting 0 */ | |
| 319 | + return s->memrefresh & 0xff; | |
| 320 | + case 0x1a: /* Auto-Refresh Interval Setting 1 */ | |
| 321 | + return s->memrefresh >> 8; | |
| 322 | + | |
| 323 | + case 0x1c: /* Power-On Sequence Timing Control */ | |
| 324 | + return s->timing[0]; | |
| 325 | + case 0x1e: /* Timing Control 0 */ | |
| 326 | + return s->timing[1]; | |
| 327 | + case 0x20: /* Timing Control 1 */ | |
| 328 | + return s->timing[2]; | |
| 329 | + | |
| 330 | + case 0x24: /* Arbitration Priority Control */ | |
| 331 | + return s->priority; | |
| 332 | + | |
| 333 | + case 0x28: /* LCD Panel Configuration */ | |
| 334 | + return s->lcd_config; | |
| 335 | + | |
| 336 | + case 0x2a: /* LCD Horizontal Display Width */ | |
| 337 | + return s->x >> 3; | |
| 338 | + case 0x2c: /* LCD Horizontal Non-display Period */ | |
| 339 | + return s->hndp; | |
| 340 | + case 0x2e: /* LCD Vertical Display Height 0 */ | |
| 341 | + return s->y & 0xff; | |
| 342 | + case 0x30: /* LCD Vertical Display Height 1 */ | |
| 343 | + return s->y >> 8; | |
| 344 | + case 0x32: /* LCD Vertical Non-display Period */ | |
| 345 | + return s->vndp; | |
| 346 | + case 0x34: /* LCD HS Pulse-width */ | |
| 347 | + return s->hsync; | |
| 348 | + case 0x36: /* LCd HS Pulse Start Position */ | |
| 349 | + return s->skipx >> 3; | |
| 350 | + case 0x38: /* LCD VS Pulse-width */ | |
| 351 | + return s->vsync; | |
| 352 | + case 0x3a: /* LCD VS Pulse Start Position */ | |
| 353 | + return s->skipy; | |
| 354 | + | |
| 355 | + case 0x3c: /* PCLK Polarity */ | |
| 356 | + return s->pclk; | |
| 357 | + | |
| 358 | + case 0x3e: /* High-speed Serial Interface Tx Configuration Port 0 */ | |
| 359 | + return s->hssi_config[0]; | |
| 360 | + case 0x40: /* High-speed Serial Interface Tx Configuration Port 1 */ | |
| 361 | + return s->hssi_config[1]; | |
| 362 | + case 0x42: /* High-speed Serial Interface Tx Mode */ | |
| 363 | + return s->hssi_config[2]; | |
| 364 | + case 0x44: /* TV Display Configuration */ | |
| 365 | + return s->tv_config; | |
| 366 | + case 0x46 ... 0x4c: /* TV Vertical Blanking Interval Data bits */ | |
| 367 | + return s->tv_timing[(reg - 0x46) >> 1]; | |
| 368 | + case 0x4e: /* VBI: Closed Caption / XDS Control / Status */ | |
| 369 | + return s->vbi; | |
| 370 | + case 0x50: /* TV Horizontal Start Position */ | |
| 371 | + return s->tv_x; | |
| 372 | + case 0x52: /* TV Vertical Start Position */ | |
| 373 | + return s->tv_y; | |
| 374 | + case 0x54: /* TV Test Pattern Setting */ | |
| 375 | + return s->tv_test; | |
| 376 | + case 0x56: /* TV Filter Setting */ | |
| 377 | + return s->tv_filter_config; | |
| 378 | + case 0x58: /* TV Filter Coefficient Index */ | |
| 379 | + return s->tv_filter_idx; | |
| 380 | + case 0x5a: /* TV Filter Coefficient Data */ | |
| 381 | + if (s->tv_filter_idx < 0x20) | |
| 382 | + return s->tv_filter_coeff[s->tv_filter_idx ++]; | |
| 383 | + return 0; | |
| 384 | + | |
| 385 | + case 0x60: /* Input YUV/RGB Translate Mode 0 */ | |
| 386 | + return s->yrc[0]; | |
| 387 | + case 0x62: /* Input YUV/RGB Translate Mode 1 */ | |
| 388 | + return s->yrc[1]; | |
| 389 | + case 0x64: /* U Data Fix */ | |
| 390 | + return s->u; | |
| 391 | + case 0x66: /* V Data Fix */ | |
| 392 | + return s->v; | |
| 393 | + | |
| 394 | + case 0x68: /* Display Mode */ | |
| 395 | + return s->mode; | |
| 396 | + | |
| 397 | + case 0x6a: /* Special Effects */ | |
| 398 | + return s->effect; | |
| 399 | + | |
| 400 | + case 0x6c: /* Input Window X Start Position 0 */ | |
| 401 | + return s->ix[0] & 0xff; | |
| 402 | + case 0x6e: /* Input Window X Start Position 1 */ | |
| 403 | + return s->ix[0] >> 3; | |
| 404 | + case 0x70: /* Input Window Y Start Position 0 */ | |
| 405 | + return s->ix[0] & 0xff; | |
| 406 | + case 0x72: /* Input Window Y Start Position 1 */ | |
| 407 | + return s->ix[0] >> 3; | |
| 408 | + case 0x74: /* Input Window X End Position 0 */ | |
| 409 | + return s->ix[1] & 0xff; | |
| 410 | + case 0x76: /* Input Window X End Position 1 */ | |
| 411 | + return s->ix[1] >> 3; | |
| 412 | + case 0x78: /* Input Window Y End Position 0 */ | |
| 413 | + return s->ix[1] & 0xff; | |
| 414 | + case 0x7a: /* Input Window Y End Position 1 */ | |
| 415 | + return s->ix[1] >> 3; | |
| 416 | + case 0x7c: /* Output Window X Start Position 0 */ | |
| 417 | + return s->ox[0] & 0xff; | |
| 418 | + case 0x7e: /* Output Window X Start Position 1 */ | |
| 419 | + return s->ox[0] >> 3; | |
| 420 | + case 0x80: /* Output Window Y Start Position 0 */ | |
| 421 | + return s->oy[0] & 0xff; | |
| 422 | + case 0x82: /* Output Window Y Start Position 1 */ | |
| 423 | + return s->oy[0] >> 3; | |
| 424 | + case 0x84: /* Output Window X End Position 0 */ | |
| 425 | + return s->ox[1] & 0xff; | |
| 426 | + case 0x86: /* Output Window X End Position 1 */ | |
| 427 | + return s->ox[1] >> 3; | |
| 428 | + case 0x88: /* Output Window Y End Position 0 */ | |
| 429 | + return s->oy[1] & 0xff; | |
| 430 | + case 0x8a: /* Output Window Y End Position 1 */ | |
| 431 | + return s->oy[1] >> 3; | |
| 432 | + | |
| 433 | + case 0x8c: /* Input Data Format */ | |
| 434 | + return s->iformat; | |
| 435 | + case 0x8e: /* Data Source Select */ | |
| 436 | + return s->source; | |
| 437 | + case 0x90: /* Display Memory Data Port */ | |
| 438 | + return 0; | |
| 439 | + | |
| 440 | + case 0xa8: /* Border Color 0 */ | |
| 441 | + return s->border_r; | |
| 442 | + case 0xaa: /* Border Color 1 */ | |
| 443 | + return s->border_g; | |
| 444 | + case 0xac: /* Border Color 2 */ | |
| 445 | + return s->border_b; | |
| 446 | + | |
| 447 | + case 0xb4: /* Gamma Correction Enable */ | |
| 448 | + return s->gamma_config; | |
| 449 | + case 0xb6: /* Gamma Correction Table Index */ | |
| 450 | + return s->gamma_idx; | |
| 451 | + case 0xb8: /* Gamma Correction Table Data */ | |
| 452 | + return s->gamma_lut[s->gamma_idx ++]; | |
| 453 | + | |
| 454 | + case 0xba: /* 3x3 Matrix Enable */ | |
| 455 | + return s->matrix_ena; | |
| 456 | + case 0xbc ... 0xde: /* Coefficient Registers */ | |
| 457 | + return s->matrix_coeff[(reg - 0xbc) >> 1]; | |
| 458 | + case 0xe0: /* 3x3 Matrix Red Offset */ | |
| 459 | + return s->matrix_r; | |
| 460 | + case 0xe2: /* 3x3 Matrix Green Offset */ | |
| 461 | + return s->matrix_g; | |
| 462 | + case 0xe4: /* 3x3 Matrix Blue Offset */ | |
| 463 | + return s->matrix_b; | |
| 464 | + | |
| 465 | + case 0xe6: /* Power-save */ | |
| 466 | + return s->pm; | |
| 467 | + case 0xe8: /* Non-display Period Control / Status */ | |
| 468 | + return s->status | (1 << 5); | |
| 469 | + case 0xea: /* RGB Interface Control */ | |
| 470 | + return s->rgbgpio_dir; | |
| 471 | + case 0xec: /* RGB Interface Status */ | |
| 472 | + return s->rgbgpio; | |
| 473 | + case 0xee: /* General-purpose IO Pins Configuration */ | |
| 474 | + return s->gpio_dir; | |
| 475 | + case 0xf0: /* General-purpose IO Pins Status / Control */ | |
| 476 | + return s->gpio; | |
| 477 | + case 0xf2: /* GPIO Positive Edge Interrupt Trigger */ | |
| 478 | + return s->gpio_edge[0]; | |
| 479 | + case 0xf4: /* GPIO Negative Edge Interrupt Trigger */ | |
| 480 | + return s->gpio_edge[1]; | |
| 481 | + case 0xf6: /* GPIO Interrupt Status */ | |
| 482 | + return s->gpio_irq; | |
| 483 | + case 0xf8: /* GPIO Pull-down Control */ | |
| 484 | + return s->gpio_pdown; | |
| 485 | + | |
| 486 | + default: | |
| 487 | + fprintf(stderr, "%s: unknown register %02x\n", __FUNCTION__, reg); | |
| 488 | + return 0; | |
| 489 | + } | |
| 490 | +} | |
| 491 | + | |
| 492 | +static void blizzard_reg_write(void *opaque, uint8_t reg, uint16_t value) | |
| 493 | +{ | |
| 494 | + struct blizzard_s *s = (struct blizzard_s *) opaque; | |
| 495 | + | |
| 496 | + switch (reg) { | |
| 497 | + case 0x04: /* PLL M-Divider */ | |
| 498 | + s->pll = (value & 0x3f) + 1; | |
| 499 | + break; | |
| 500 | + case 0x06: /* PLL Lock Range Control */ | |
| 501 | + s->pll_range = value & 3; | |
| 502 | + break; | |
| 503 | + case 0x08: /* PLL Lock Synthesis Control 0 */ | |
| 504 | + s->pll_ctrl &= 0xf00; | |
| 505 | + s->pll_ctrl |= (value << 0) & 0x0ff; | |
| 506 | + break; | |
| 507 | + case 0x0a: /* PLL Lock Synthesis Control 1 */ | |
| 508 | + s->pll_ctrl &= 0x0ff; | |
| 509 | + s->pll_ctrl |= (value << 8) & 0xf00; | |
| 510 | + break; | |
| 511 | + case 0x0c: /* PLL Mode Control 0 */ | |
| 512 | + s->pll_mode = value & 0x77; | |
| 513 | + if ((value & 3) == 0 || (value & 3) == 3) | |
| 514 | + fprintf(stderr, "%s: wrong PLL Control bits (%i)\n", | |
| 515 | + __FUNCTION__, value & 3); | |
| 516 | + break; | |
| 517 | + | |
| 518 | + case 0x0e: /* Clock-Source Select */ | |
| 519 | + s->clksel = value & 0xff; | |
| 520 | + break; | |
| 521 | + | |
| 522 | + case 0x10: /* Memory Controller Activate */ | |
| 523 | + s->memenable = value & 1; | |
| 524 | + break; | |
| 525 | + case 0x14: /* Memory Controller Bank 0 Status Flag */ | |
| 526 | + break; | |
| 527 | + | |
| 528 | + case 0x18: /* Auto-Refresh Interval Setting 0 */ | |
| 529 | + s->memrefresh &= 0xf00; | |
| 530 | + s->memrefresh |= (value << 0) & 0x0ff; | |
| 531 | + break; | |
| 532 | + case 0x1a: /* Auto-Refresh Interval Setting 1 */ | |
| 533 | + s->memrefresh &= 0x0ff; | |
| 534 | + s->memrefresh |= (value << 8) & 0xf00; | |
| 535 | + break; | |
| 536 | + | |
| 537 | + case 0x1c: /* Power-On Sequence Timing Control */ | |
| 538 | + s->timing[0] = value & 0x7f; | |
| 539 | + break; | |
| 540 | + case 0x1e: /* Timing Control 0 */ | |
| 541 | + s->timing[1] = value & 0x17; | |
| 542 | + break; | |
| 543 | + case 0x20: /* Timing Control 1 */ | |
| 544 | + s->timing[2] = value & 0x35; | |
| 545 | + break; | |
| 546 | + | |
| 547 | + case 0x24: /* Arbitration Priority Control */ | |
| 548 | + s->priority = value & 1; | |
| 549 | + break; | |
| 550 | + | |
| 551 | + case 0x28: /* LCD Panel Configuration */ | |
| 552 | + s->lcd_config = value & 0xff; | |
| 553 | + if (value & (1 << 7)) | |
| 554 | + fprintf(stderr, "%s: data swap not supported!\n", __FUNCTION__); | |
| 555 | + break; | |
| 556 | + | |
| 557 | + case 0x2a: /* LCD Horizontal Display Width */ | |
| 558 | + s->x = value << 3; | |
| 559 | + break; | |
| 560 | + case 0x2c: /* LCD Horizontal Non-display Period */ | |
| 561 | + s->hndp = value & 0xff; | |
| 562 | + break; | |
| 563 | + case 0x2e: /* LCD Vertical Display Height 0 */ | |
| 564 | + s->y &= 0x300; | |
| 565 | + s->y |= (value << 0) & 0x0ff; | |
| 566 | + break; | |
| 567 | + case 0x30: /* LCD Vertical Display Height 1 */ | |
| 568 | + s->y &= 0x0ff; | |
| 569 | + s->y |= (value << 8) & 0x300; | |
| 570 | + break; | |
| 571 | + case 0x32: /* LCD Vertical Non-display Period */ | |
| 572 | + s->vndp = value & 0xff; | |
| 573 | + break; | |
| 574 | + case 0x34: /* LCD HS Pulse-width */ | |
| 575 | + s->hsync = value & 0xff; | |
| 576 | + break; | |
| 577 | + case 0x36: /* LCD HS Pulse Start Position */ | |
| 578 | + s->skipx = value & 0xff; | |
| 579 | + break; | |
| 580 | + case 0x38: /* LCD VS Pulse-width */ | |
| 581 | + s->vsync = value & 0xbf; | |
| 582 | + break; | |
| 583 | + case 0x3a: /* LCD VS Pulse Start Position */ | |
| 584 | + s->skipy = value & 0xff; | |
| 585 | + break; | |
| 586 | + | |
| 587 | + case 0x3c: /* PCLK Polarity */ | |
| 588 | + s->pclk = value & 0x82; | |
| 589 | + /* Affects calculation of s->hndp, s->hsync and s->skipx. */ | |
| 590 | + break; | |
| 591 | + | |
| 592 | + case 0x3e: /* High-speed Serial Interface Tx Configuration Port 0 */ | |
| 593 | + s->hssi_config[0] = value; | |
| 594 | + break; | |
| 595 | + case 0x40: /* High-speed Serial Interface Tx Configuration Port 1 */ | |
| 596 | + s->hssi_config[1] = value; | |
| 597 | + if (((value >> 4) & 3) == 3) | |
| 598 | + fprintf(stderr, "%s: Illegal active-data-links value\n", | |
| 599 | + __FUNCTION__); | |
| 600 | + break; | |
| 601 | + case 0x42: /* High-speed Serial Interface Tx Mode */ | |
| 602 | + s->hssi_config[2] = value & 0xbd; | |
| 603 | + break; | |
| 604 | + | |
| 605 | + case 0x44: /* TV Display Configuration */ | |
| 606 | + s->tv_config = value & 0xfe; | |
| 607 | + break; | |
| 608 | + case 0x46 ... 0x4c: /* TV Vertical Blanking Interval Data bits 0 */ | |
| 609 | + s->tv_timing[(reg - 0x46) >> 1] = value; | |
| 610 | + break; | |
| 611 | + case 0x4e: /* VBI: Closed Caption / XDS Control / Status */ | |
| 612 | + s->vbi = value; | |
| 613 | + break; | |
| 614 | + case 0x50: /* TV Horizontal Start Position */ | |
| 615 | + s->tv_x = value; | |
| 616 | + break; | |
| 617 | + case 0x52: /* TV Vertical Start Position */ | |
| 618 | + s->tv_y = value & 0x7f; | |
| 619 | + break; | |
| 620 | + case 0x54: /* TV Test Pattern Setting */ | |
| 621 | + s->tv_test = value; | |
| 622 | + break; | |
| 623 | + case 0x56: /* TV Filter Setting */ | |
| 624 | + s->tv_filter_config = value & 0xbf; | |
| 625 | + break; | |
| 626 | + case 0x58: /* TV Filter Coefficient Index */ | |
| 627 | + s->tv_filter_idx = value & 0x1f; | |
| 628 | + break; | |
| 629 | + case 0x5a: /* TV Filter Coefficient Data */ | |
| 630 | + if (s->tv_filter_idx < 0x20) | |
| 631 | + s->tv_filter_coeff[s->tv_filter_idx ++] = value; | |
| 632 | + break; | |
| 633 | + | |
| 634 | + case 0x60: /* Input YUV/RGB Translate Mode 0 */ | |
| 635 | + s->yrc[0] = value & 0xb0; | |
| 636 | + break; | |
| 637 | + case 0x62: /* Input YUV/RGB Translate Mode 1 */ | |
| 638 | + s->yrc[1] = value & 0x30; | |
| 639 | + break; | |
| 640 | + case 0x64: /* U Data Fix */ | |
| 641 | + s->u = value & 0xff; | |
| 642 | + break; | |
| 643 | + case 0x66: /* V Data Fix */ | |
| 644 | + s->v = value & 0xff; | |
| 645 | + break; | |
| 646 | + | |
| 647 | + case 0x68: /* Display Mode */ | |
| 648 | + if ((s->mode ^ value) & 3) | |
| 649 | + s->invalidate = 1; | |
| 650 | + s->mode = value & 0xb7; | |
| 651 | + s->enable = value & 1; | |
| 652 | + s->blank = (value >> 1) & 1; | |
| 653 | + if (value & (1 << 4)) | |
| 654 | + fprintf(stderr, "%s: Macrovision enable attempt!\n", __FUNCTION__); | |
| 655 | + break; | |
| 656 | + | |
| 657 | + case 0x6a: /* Special Effects */ | |
| 658 | + s->effect = value & 0xfb; | |
| 659 | + break; | |
| 660 | + | |
| 661 | + case 0x6c: /* Input Window X Start Position 0 */ | |
| 662 | + s->ix[0] &= 0x300; | |
| 663 | + s->ix[0] |= (value << 0) & 0x0ff; | |
| 664 | + break; | |
| 665 | + case 0x6e: /* Input Window X Start Position 1 */ | |
| 666 | + s->ix[0] &= 0x0ff; | |
| 667 | + s->ix[0] |= (value << 8) & 0x300; | |
| 668 | + break; | |
| 669 | + case 0x70: /* Input Window Y Start Position 0 */ | |
| 670 | + s->iy[0] &= 0x300; | |
| 671 | + s->iy[0] |= (value << 0) & 0x0ff; | |
| 672 | + break; | |
| 673 | + case 0x72: /* Input Window Y Start Position 1 */ | |
| 674 | + s->iy[0] &= 0x0ff; | |
| 675 | + s->iy[0] |= (value << 8) & 0x300; | |
| 676 | + break; | |
| 677 | + case 0x74: /* Input Window X End Position 0 */ | |
| 678 | + s->ix[1] &= 0x300; | |
| 679 | + s->ix[1] |= (value << 0) & 0x0ff; | |
| 680 | + break; | |
| 681 | + case 0x76: /* Input Window X End Position 1 */ | |
| 682 | + s->ix[1] &= 0x0ff; | |
| 683 | + s->ix[1] |= (value << 8) & 0x300; | |
| 684 | + break; | |
| 685 | + case 0x78: /* Input Window Y End Position 0 */ | |
| 686 | + s->iy[1] &= 0x300; | |
| 687 | + s->iy[1] |= (value << 0) & 0x0ff; | |
| 688 | + break; | |
| 689 | + case 0x7a: /* Input Window Y End Position 1 */ | |
| 690 | + s->iy[1] &= 0x0ff; | |
| 691 | + s->iy[1] |= (value << 8) & 0x300; | |
| 692 | + break; | |
| 693 | + case 0x7c: /* Output Window X Start Position 0 */ | |
| 694 | + s->ox[0] &= 0x300; | |
| 695 | + s->ox[0] |= (value << 0) & 0x0ff; | |
| 696 | + break; | |
| 697 | + case 0x7e: /* Output Window X Start Position 1 */ | |
| 698 | + s->ox[0] &= 0x0ff; | |
| 699 | + s->ox[0] |= (value << 8) & 0x300; | |
| 700 | + break; | |
| 701 | + case 0x80: /* Output Window Y Start Position 0 */ | |
| 702 | + s->oy[0] &= 0x300; | |
| 703 | + s->oy[0] |= (value << 0) & 0x0ff; | |
| 704 | + break; | |
| 705 | + case 0x82: /* Output Window Y Start Position 1 */ | |
| 706 | + s->oy[0] &= 0x0ff; | |
| 707 | + s->oy[0] |= (value << 8) & 0x300; | |
| 708 | + break; | |
| 709 | + case 0x84: /* Output Window X End Position 0 */ | |
| 710 | + s->ox[1] &= 0x300; | |
| 711 | + s->ox[1] |= (value << 0) & 0x0ff; | |
| 712 | + break; | |
| 713 | + case 0x86: /* Output Window X End Position 1 */ | |
| 714 | + s->ox[1] &= 0x0ff; | |
| 715 | + s->ox[1] |= (value << 8) & 0x300; | |
| 716 | + break; | |
| 717 | + case 0x88: /* Output Window Y End Position 0 */ | |
| 718 | + s->oy[1] &= 0x300; | |
| 719 | + s->oy[1] |= (value << 0) & 0x0ff; | |
| 720 | + break; | |
| 721 | + case 0x8a: /* Output Window Y End Position 1 */ | |
| 722 | + s->oy[1] &= 0x0ff; | |
| 723 | + s->oy[1] |= (value << 8) & 0x300; | |
| 724 | + break; | |
| 725 | + | |
| 726 | + case 0x8c: /* Input Data Format */ | |
| 727 | + s->iformat = value & 0xf; | |
| 728 | + s->bpp = blizzard_iformat_bpp[s->iformat]; | |
| 729 | + if (!s->bpp) | |
| 730 | + fprintf(stderr, "%s: Illegal or unsupported input format %x\n", | |
| 731 | + __FUNCTION__, s->iformat); | |
| 732 | + break; | |
| 733 | + case 0x8e: /* Data Source Select */ | |
| 734 | + s->source = value & 7; | |
| 735 | + /* Currently all windows will be "destructive overlays". */ | |
| 736 | + if ((!(s->effect & (1 << 3)) && (s->ix[0] != s->ox[0] || | |
| 737 | + s->iy[0] != s->oy[0] || | |
| 738 | + s->ix[1] != s->ox[1] || | |
| 739 | + s->iy[1] != s->oy[1])) || | |
| 740 | + !((s->ix[1] - s->ix[0]) & (s->iy[1] - s->iy[0]) & | |
| 741 | + (s->ox[1] - s->ox[0]) & (s->oy[1] - s->oy[0]) & 1)) | |
| 742 | + fprintf(stderr, "%s: Illegal input/output window positions\n", | |
| 743 | + __FUNCTION__); | |
| 744 | + | |
| 745 | + blizzard_transfer_setup(s); | |
| 746 | + break; | |
| 747 | + | |
| 748 | + case 0x90: /* Display Memory Data Port */ | |
| 749 | + if (!s->data.len && !blizzard_transfer_setup(s)) | |
| 750 | + break; | |
| 751 | + | |
| 752 | + *s->data.ptr ++ = value; | |
| 753 | + if (-- s->data.len == 0) | |
| 754 | + blizzard_window(s); | |
| 755 | + break; | |
| 756 | + | |
| 757 | + case 0xa8: /* Border Color 0 */ | |
| 758 | + s->border_r = value; | |
| 759 | + break; | |
| 760 | + case 0xaa: /* Border Color 1 */ | |
| 761 | + s->border_g = value; | |
| 762 | + break; | |
| 763 | + case 0xac: /* Border Color 2 */ | |
| 764 | + s->border_b = value; | |
| 765 | + break; | |
| 766 | + | |
| 767 | + case 0xb4: /* Gamma Correction Enable */ | |
| 768 | + s->gamma_config = value & 0x87; | |
| 769 | + break; | |
| 770 | + case 0xb6: /* Gamma Correction Table Index */ | |
| 771 | + s->gamma_idx = value; | |
| 772 | + break; | |
| 773 | + case 0xb8: /* Gamma Correction Table Data */ | |
| 774 | + s->gamma_lut[s->gamma_idx ++] = value; | |
| 775 | + break; | |
| 776 | + | |
| 777 | + case 0xba: /* 3x3 Matrix Enable */ | |
| 778 | + s->matrix_ena = value & 1; | |
| 779 | + break; | |
| 780 | + case 0xbc ... 0xde: /* Coefficient Registers */ | |
| 781 | + s->matrix_coeff[(reg - 0xbc) >> 1] = value & ((reg & 2) ? 0x80 : 0xff); | |
| 782 | + break; | |
| 783 | + case 0xe0: /* 3x3 Matrix Red Offset */ | |
| 784 | + s->matrix_r = value; | |
| 785 | + break; | |
| 786 | + case 0xe2: /* 3x3 Matrix Green Offset */ | |
| 787 | + s->matrix_g = value; | |
| 788 | + break; | |
| 789 | + case 0xe4: /* 3x3 Matrix Blue Offset */ | |
| 790 | + s->matrix_b = value; | |
| 791 | + break; | |
| 792 | + | |
| 793 | + case 0xe6: /* Power-save */ | |
| 794 | + s->pm = value & 0x83; | |
| 795 | + if (value & s->mode & 1) | |
| 796 | + fprintf(stderr, "%s: The display must be disabled before entering " | |
| 797 | + "Standby Mode\n", __FUNCTION__); | |
| 798 | + break; | |
| 799 | + case 0xe8: /* Non-display Period Control / Status */ | |
| 800 | + s->status = value & 0x1b; | |
| 801 | + break; | |
| 802 | + case 0xea: /* RGB Interface Control */ | |
| 803 | + s->rgbgpio_dir = value & 0x8f; | |
| 804 | + break; | |
| 805 | + case 0xec: /* RGB Interface Status */ | |
| 806 | + s->rgbgpio = value & 0xcf; | |
| 807 | + break; | |
| 808 | + case 0xee: /* General-purpose IO Pins Configuration */ | |
| 809 | + s->gpio_dir = value; | |
| 810 | + break; | |
| 811 | + case 0xf0: /* General-purpose IO Pins Status / Control */ | |
| 812 | + s->gpio = value; | |
| 813 | + break; | |
| 814 | + case 0xf2: /* GPIO Positive Edge Interrupt Trigger */ | |
| 815 | + s->gpio_edge[0] = value; | |
| 816 | + break; | |
| 817 | + case 0xf4: /* GPIO Negative Edge Interrupt Trigger */ | |
| 818 | + s->gpio_edge[1] = value; | |
| 819 | + break; | |
| 820 | + case 0xf6: /* GPIO Interrupt Status */ | |
| 821 | + s->gpio_irq &= value; | |
| 822 | + break; | |
| 823 | + case 0xf8: /* GPIO Pull-down Control */ | |
| 824 | + s->gpio_pdown = value; | |
| 825 | + break; | |
| 826 | + | |
| 827 | + default: | |
| 828 | + fprintf(stderr, "%s: unknown register %02x\n", __FUNCTION__, reg); | |
| 829 | + break; | |
| 830 | + } | |
| 831 | +} | |
| 832 | + | |
| 833 | +uint16_t s1d13745_read(void *opaque, int dc) | |
| 834 | +{ | |
| 835 | + struct blizzard_s *s = (struct blizzard_s *) opaque; | |
| 836 | + uint16_t value = blizzard_reg_read(s, s->reg); | |
| 837 | + | |
| 838 | + if (s->swallow -- > 0) | |
| 839 | + return 0; | |
| 840 | + if (dc) | |
| 841 | + s->reg ++; | |
| 842 | + | |
| 843 | + return value; | |
| 844 | +} | |
| 845 | + | |
| 846 | +void s1d13745_write(void *opaque, int dc, uint16_t value) | |
| 847 | +{ | |
| 848 | + struct blizzard_s *s = (struct blizzard_s *) opaque; | |
| 849 | + | |
| 850 | + if (s->swallow -- > 0) | |
| 851 | + return; | |
| 852 | + if (dc) { | |
| 853 | + blizzard_reg_write(s, s->reg, value); | |
| 854 | + | |
| 855 | + if (s->reg != 0x90 && s->reg != 0x5a && s->reg != 0xb8) | |
| 856 | + s->reg += 2; | |
| 857 | + } else | |
| 858 | + s->reg = value & 0xff; | |
| 859 | +} | |
| 860 | + | |
| 861 | +void s1d13745_write_block(void *opaque, int dc, | |
| 862 | + void *buf, size_t len, int pitch) | |
| 863 | +{ | |
| 864 | + struct blizzard_s *s = (struct blizzard_s *) opaque; | |
| 865 | + | |
| 866 | + while (len > 0) { | |
| 867 | + if (s->reg == 0x90 && dc && | |
| 868 | + (s->data.len || blizzard_transfer_setup(s)) && | |
| 869 | + len >= (s->data.len << 1)) { | |
| 870 | + len -= s->data.len << 1; | |
| 871 | + s->data.len = 0; | |
| 872 | + s->data.data = buf; | |
| 873 | + if (pitch) | |
| 874 | + s->data.pitch = pitch; | |
| 875 | + blizzard_window(s); | |
| 876 | + s->data.data = s->data.buf; | |
| 877 | + continue; | |
| 878 | + } | |
| 879 | + | |
| 880 | + s1d13745_write(opaque, dc, *(uint16_t *) buf); | |
| 881 | + len -= 2; | |
| 882 | + buf += 2; | |
| 883 | + } | |
| 884 | + | |
| 885 | + return; | |
| 886 | +} | |
| 887 | + | |
| 888 | +static void blizzard_update_display(void *opaque) | |
| 889 | +{ | |
| 890 | + struct blizzard_s *s = (struct blizzard_s *) opaque; | |
| 891 | + int y, bypp, bypl, bwidth; | |
| 892 | + uint8_t *src, *dst; | |
| 893 | + | |
| 894 | + if (!s->enable) | |
| 895 | + return; | |
| 896 | + | |
| 897 | + if (s->x != s->state->width || s->y != s->state->height) { | |
| 898 | + s->invalidate = 1; | |
| 899 | + dpy_resize(s->state, s->x, s->y); | |
| 900 | + } | |
| 901 | + | |
| 902 | + if (s->invalidate) { | |
| 903 | + s->invalidate = 0; | |
| 904 | + | |
| 905 | + if (s->blank) { | |
| 906 | + bypp = (s->state->depth + 7) >> 3; | |
| 907 | + memset(s->state->data, 0, bypp * s->x * s->y); | |
| 908 | + return; | |
| 909 | + } | |
| 910 | + | |
| 911 | + s->mx[0] = 0; | |
| 912 | + s->mx[1] = s->x; | |
| 913 | + s->my[0] = 0; | |
| 914 | + s->my[1] = s->y; | |
| 915 | + } | |
| 916 | + | |
| 917 | + if (s->mx[1] <= s->mx[0]) | |
| 918 | + return; | |
| 919 | + | |
| 920 | + bypp = (s->state->depth + 7) >> 3; | |
| 921 | + bypl = bypp * s->x; | |
| 922 | + bwidth = bypp * (s->mx[1] - s->mx[0]); | |
| 923 | + y = s->my[0]; | |
| 924 | + src = s->fb + bypl * y + bypp * s->mx[0]; | |
| 925 | + dst = s->state->data + bypl * y + bypp * s->mx[0]; | |
| 926 | + for (; y < s->my[1]; y ++, src += bypl, dst += bypl) | |
| 927 | + memcpy(dst, src, bwidth); | |
| 928 | + | |
| 929 | + dpy_update(s->state, s->mx[0], s->my[0], | |
| 930 | + s->mx[1] - s->mx[0], y - s->my[0]); | |
| 931 | + | |
| 932 | + s->mx[0] = s->x; | |
| 933 | + s->mx[1] = 0; | |
| 934 | + s->my[0] = s->y; | |
| 935 | + s->my[1] = 0; | |
| 936 | +} | |
| 937 | + | |
| 938 | +static void blizzard_screen_dump(void *opaque, const char *filename) { | |
| 939 | + struct blizzard_s *s = (struct blizzard_s *) opaque; | |
| 940 | + | |
| 941 | + blizzard_update_display(opaque); | |
| 942 | + if (s && s->state->data) | |
| 943 | + ppm_save(filename, s->state->data, s->x, s->y, s->state->linesize); | |
| 944 | +} | |
| 945 | + | |
| 946 | +#define DEPTH 8 | |
| 947 | +#include "blizzard_template.h" | |
| 948 | +#define DEPTH 15 | |
| 949 | +#include "blizzard_template.h" | |
| 950 | +#define DEPTH 16 | |
| 951 | +#include "blizzard_template.h" | |
| 952 | +#define DEPTH 24 | |
| 953 | +#include "blizzard_template.h" | |
| 954 | +#define DEPTH 32 | |
| 955 | +#include "blizzard_template.h" | |
| 956 | + | |
| 957 | +void *s1d13745_init(qemu_irq gpio_int, DisplayState *ds) | |
| 958 | +{ | |
| 959 | + struct blizzard_s *s = (struct blizzard_s *) qemu_mallocz(sizeof(*s)); | |
| 960 | + | |
| 961 | + s->state = ds; | |
| 962 | + s->fb = qemu_malloc(0x180000); | |
| 963 | + | |
| 964 | + switch (s->state->depth) { | |
| 965 | + case 0: | |
| 966 | + s->line_fn_tab[0] = s->line_fn_tab[1] = | |
| 967 | + qemu_mallocz(sizeof(blizzard_fn_t) * 0x10); | |
| 968 | + break; | |
| 969 | + case 8: | |
| 970 | + s->line_fn_tab[0] = blizzard_draw_fn_8; | |
| 971 | + s->line_fn_tab[1] = blizzard_draw_fn_r_8; | |
| 972 | + break; | |
| 973 | + case 15: | |
| 974 | + s->line_fn_tab[0] = blizzard_draw_fn_15; | |
| 975 | + s->line_fn_tab[1] = blizzard_draw_fn_r_15; | |
| 976 | + break; | |
| 977 | + case 16: | |
| 978 | + s->line_fn_tab[0] = blizzard_draw_fn_16; | |
| 979 | + s->line_fn_tab[1] = blizzard_draw_fn_r_16; | |
| 980 | + break; | |
| 981 | + case 24: | |
| 982 | + s->line_fn_tab[0] = blizzard_draw_fn_24; | |
| 983 | + s->line_fn_tab[1] = blizzard_draw_fn_r_24; | |
| 984 | + break; | |
| 985 | + case 32: | |
| 986 | + s->line_fn_tab[0] = blizzard_draw_fn_32; | |
| 987 | + s->line_fn_tab[1] = blizzard_draw_fn_r_32; | |
| 988 | + break; | |
| 989 | + default: | |
| 990 | + fprintf(stderr, "%s: Bad color depth\n", __FUNCTION__); | |
| 991 | + exit(1); | |
| 992 | + } | |
| 993 | + | |
| 994 | + blizzard_reset(s); | |
| 995 | + | |
| 996 | + graphic_console_init(s->state, blizzard_update_display, | |
| 997 | + blizzard_invalidate_display, blizzard_screen_dump, | |
| 998 | + NULL, s); | |
| 999 | + | |
| 1000 | + return s; | |
| 1001 | +} | ... | ... |
hw/blizzard_template.h
0 → 100644
| 1 | +/* | |
| 2 | + * QEMU Epson S1D13744/S1D13745 templates | |
| 3 | + * | |
| 4 | + * Copyright (C) 2008 Nokia Corporation | |
| 5 | + * Written by Andrzej Zaborowski <andrew@openedhand.com> | |
| 6 | + * | |
| 7 | + * This program is free software; you can redistribute it and/or | |
| 8 | + * modify it under the terms of the GNU General Public License as | |
| 9 | + * published by the Free Software Foundation; either version 2 or | |
| 10 | + * (at your option) version 3 of the License. | |
| 11 | + * | |
| 12 | + * This program is distributed in the hope that it will be useful, | |
| 13 | + * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
| 14 | + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
| 15 | + * GNU General Public License for more details. | |
| 16 | + * | |
| 17 | + * You should have received a copy of the GNU General Public License | |
| 18 | + * along with this program; if not, write to the Free Software | |
| 19 | + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, | |
| 20 | + * MA 02111-1307 USA | |
| 21 | + */ | |
| 22 | + | |
| 23 | +#define SKIP_PIXEL(to) to += deststep | |
| 24 | +#if DEPTH == 8 | |
| 25 | +# define PIXEL_TYPE uint8_t | |
| 26 | +# define COPY_PIXEL(to, from) *to = from; SKIP_PIXEL(to) | |
| 27 | +# define COPY_PIXEL1(to, from) *to ++ = from | |
| 28 | +#elif DEPTH == 15 || DEPTH == 16 | |
| 29 | +# define PIXEL_TYPE uint16_t | |
| 30 | +# define COPY_PIXEL(to, from) *to = from; SKIP_PIXEL(to) | |
| 31 | +# define COPY_PIXEL1(to, from) *to ++ = from | |
| 32 | +#elif DEPTH == 24 | |
| 33 | +# define PIXEL_TYPE uint8_t | |
| 34 | +# define COPY_PIXEL(to, from) \ | |
| 35 | + to[0] = from; to[1] = (from) >> 8; to[2] = (from) >> 16; SKIP_PIXEL(to) | |
| 36 | +# define COPY_PIXEL1(to, from) \ | |
| 37 | + *to ++ = from; *to ++ = (from) >> 8; *to ++ = (from) >> 16 | |
| 38 | +#elif DEPTH == 32 | |
| 39 | +# define PIXEL_TYPE uint32_t | |
| 40 | +# define COPY_PIXEL(to, from) *to = from; SKIP_PIXEL(to) | |
| 41 | +# define COPY_PIXEL1(to, from) *to ++ = from | |
| 42 | +#else | |
| 43 | +# error unknown bit depth | |
| 44 | +#endif | |
| 45 | + | |
| 46 | +#ifdef WORDS_BIGENDIAN | |
| 47 | +# define SWAP_WORDS 1 | |
| 48 | +#endif | |
| 49 | + | |
| 50 | +static void glue(blizzard_draw_line16_, DEPTH)(PIXEL_TYPE *dest, | |
| 51 | + const uint16_t *src, unsigned int width) | |
| 52 | +{ | |
| 53 | +#if !defined(SWAP_WORDS) && DEPTH == 16 | |
| 54 | + memcpy(dest, src, width << 1); | |
| 55 | +#else | |
| 56 | + uint16_t data; | |
| 57 | + unsigned int r, g, b; | |
| 58 | + const uint16_t *end = (void *) src + width; | |
| 59 | + while (src < end) { | |
| 60 | + data = lduw_raw(src ++); | |
| 61 | + b = (data & 0x1f) << 3; | |
| 62 | + data >>= 5; | |
| 63 | + g = (data & 0x3f) << 2; | |
| 64 | + data >>= 6; | |
| 65 | + r = (data & 0x1f) << 3; | |
| 66 | + data >>= 5; | |
| 67 | + COPY_PIXEL1(dest, glue(rgb_to_pixel, DEPTH)(r, g, b)); | |
| 68 | + } | |
| 69 | +#endif | |
| 70 | +} | |
| 71 | + | |
| 72 | +static void glue(blizzard_draw_line24mode1_, DEPTH)(PIXEL_TYPE *dest, | |
| 73 | + const uint8_t *src, unsigned int width) | |
| 74 | +{ | |
| 75 | + /* TODO: check if SDL 24-bit planes are not in the same format and | |
| 76 | + * if so, use memcpy */ | |
| 77 | + unsigned int r[2], g[2], b[2]; | |
| 78 | + const uint8_t *end = src + width; | |
| 79 | + while (src < end) { | |
| 80 | + g[0] = *src ++; | |
| 81 | + r[0] = *src ++; | |
| 82 | + r[1] = *src ++; | |
| 83 | + b[0] = *src ++; | |
| 84 | + COPY_PIXEL1(dest, glue(rgb_to_pixel, DEPTH)(r[0], g[0], b[0])); | |
| 85 | + b[1] = *src ++; | |
| 86 | + g[1] = *src ++; | |
| 87 | + COPY_PIXEL1(dest, glue(rgb_to_pixel, DEPTH)(r[1], g[1], b[1])); | |
| 88 | + } | |
| 89 | +} | |
| 90 | + | |
| 91 | +static void glue(blizzard_draw_line24mode2_, DEPTH)(PIXEL_TYPE *dest, | |
| 92 | + const uint8_t *src, unsigned int width) | |
| 93 | +{ | |
| 94 | + unsigned int r, g, b; | |
| 95 | + const uint8_t *end = src + width; | |
| 96 | + while (src < end) { | |
| 97 | + r = *src ++; | |
| 98 | + src ++; | |
| 99 | + b = *src ++; | |
| 100 | + g = *src ++; | |
| 101 | + COPY_PIXEL1(dest, glue(rgb_to_pixel, DEPTH)(r, g, b)); | |
| 102 | + } | |
| 103 | +} | |
| 104 | + | |
| 105 | +/* No rotation */ | |
| 106 | +static blizzard_fn_t glue(blizzard_draw_fn_, DEPTH)[0x10] = { | |
| 107 | + NULL, | |
| 108 | + /* RGB 5:6:5*/ | |
| 109 | + (blizzard_fn_t) glue(blizzard_draw_line16_, DEPTH), | |
| 110 | + /* RGB 6:6:6 mode 1 */ | |
| 111 | + (blizzard_fn_t) glue(blizzard_draw_line24mode1_, DEPTH), | |
| 112 | + /* RGB 8:8:8 mode 1 */ | |
| 113 | + (blizzard_fn_t) glue(blizzard_draw_line24mode1_, DEPTH), | |
| 114 | + NULL, NULL, | |
| 115 | + /* RGB 6:6:6 mode 2 */ | |
| 116 | + (blizzard_fn_t) glue(blizzard_draw_line24mode2_, DEPTH), | |
| 117 | + /* RGB 8:8:8 mode 2 */ | |
| 118 | + (blizzard_fn_t) glue(blizzard_draw_line24mode2_, DEPTH), | |
| 119 | + /* YUV 4:2:2 */ | |
| 120 | + NULL, | |
| 121 | + /* YUV 4:2:0 */ | |
| 122 | + NULL, | |
| 123 | + NULL, NULL, NULL, NULL, NULL, NULL, | |
| 124 | +}; | |
| 125 | + | |
| 126 | +/* 90deg, 180deg and 270deg rotation */ | |
| 127 | +static blizzard_fn_t glue(blizzard_draw_fn_r_, DEPTH)[0x10] = { | |
| 128 | + /* TODO */ | |
| 129 | + [0 ... 0xf] = NULL, | |
| 130 | +}; | |
| 131 | + | |
| 132 | +#undef DEPTH | |
| 133 | +#undef SKIP_PIXEL | |
| 134 | +#undef COPY_PIXEL | |
| 135 | +#undef COPY_PIXEL1 | |
| 136 | +#undef PIXEL_TYPE | |
| 137 | + | |
| 138 | +#undef SWAP_WORDS | ... | ... |
hw/boards.h
| ... | ... | @@ -81,6 +81,9 @@ extern QEMUMachine terrierpda_machine; |
| 81 | 81 | /* palm.c */ |
| 82 | 82 | extern QEMUMachine palmte_machine; |
| 83 | 83 | |
| 84 | +/* nseries.c */ | |
| 85 | +extern QEMUMachine n800_machine; | |
| 86 | + | |
| 84 | 87 | /* gumstix.c */ |
| 85 | 88 | extern QEMUMachine connex_machine; |
| 86 | 89 | extern QEMUMachine verdex_machine; | ... | ... |
hw/cbus.c
0 → 100644
| 1 | +/* | |
| 2 | + * CBUS three-pin bus and the Retu / Betty / Tahvo / Vilma / Avilma / | |
| 3 | + * Hinku / Vinku / Ahne / Pihi chips used in various Nokia platforms. | |
| 4 | + * Based on reverse-engineering of a linux driver. | |
| 5 | + * | |
| 6 | + * Copyright (C) 2008 Nokia Corporation | |
| 7 | + * Written by Andrzej Zaborowski <andrew@openedhand.com> | |
| 8 | + * | |
| 9 | + * This program is free software; you can redistribute it and/or | |
| 10 | + * modify it under the terms of the GNU General Public License as | |
| 11 | + * published by the Free Software Foundation; either version 2 or | |
| 12 | + * (at your option) version 3 of the License. | |
| 13 | + * | |
| 14 | + * This program is distributed in the hope that it will be useful, | |
| 15 | + * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
| 16 | + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
| 17 | + * GNU General Public License for more details. | |
| 18 | + * | |
| 19 | + * You should have received a copy of the GNU General Public License | |
| 20 | + * along with this program; if not, write to the Free Software | |
| 21 | + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, | |
| 22 | + * MA 02111-1307 USA | |
| 23 | + */ | |
| 24 | + | |
| 25 | +#include "qemu-common.h" | |
| 26 | +#include "irq.h" | |
| 27 | +#include "devices.h" | |
| 28 | +#include "sysemu.h" | |
| 29 | + | |
| 30 | +//#define DEBUG | |
| 31 | + | |
| 32 | +struct cbus_slave_s; | |
| 33 | +struct cbus_priv_s { | |
| 34 | + struct cbus_s cbus; | |
| 35 | + | |
| 36 | + int sel; | |
| 37 | + int dat; | |
| 38 | + int clk; | |
| 39 | + int bit; | |
| 40 | + int dir; | |
| 41 | + uint16_t val; | |
| 42 | + qemu_irq dat_out; | |
| 43 | + | |
| 44 | + int addr; | |
| 45 | + int reg; | |
| 46 | + int rw; | |
| 47 | + enum { | |
| 48 | + cbus_address, | |
| 49 | + cbus_value, | |
| 50 | + } cycle; | |
| 51 | + | |
| 52 | + struct cbus_slave_s *slave[8]; | |
| 53 | +}; | |
| 54 | + | |
| 55 | +struct cbus_slave_s { | |
| 56 | + void *opaque; | |
| 57 | + void (*io)(void *opaque, int rw, int reg, uint16_t *val); | |
| 58 | + int addr; | |
| 59 | +}; | |
| 60 | + | |
| 61 | +static void cbus_io(struct cbus_priv_s *s) | |
| 62 | +{ | |
| 63 | + if (s->slave[s->addr]) | |
| 64 | + s->slave[s->addr]->io(s->slave[s->addr]->opaque, | |
| 65 | + s->rw, s->reg, &s->val); | |
| 66 | + else | |
| 67 | + cpu_abort(cpu_single_env, "%s: bad slave address %i\n", | |
| 68 | + __FUNCTION__, s->addr); | |
| 69 | +} | |
| 70 | + | |
| 71 | +static void cbus_cycle(struct cbus_priv_s *s) | |
| 72 | +{ | |
| 73 | + switch (s->cycle) { | |
| 74 | + case cbus_address: | |
| 75 | + s->addr = (s->val >> 6) & 7; | |
| 76 | + s->rw = (s->val >> 5) & 1; | |
| 77 | + s->reg = (s->val >> 0) & 0x1f; | |
| 78 | + | |
| 79 | + s->cycle = cbus_value; | |
| 80 | + s->bit = 15; | |
| 81 | + s->dir = !s->rw; | |
| 82 | + s->val = 0; | |
| 83 | + | |
| 84 | + if (s->rw) | |
| 85 | + cbus_io(s); | |
| 86 | + break; | |
| 87 | + | |
| 88 | + case cbus_value: | |
| 89 | + if (!s->rw) | |
| 90 | + cbus_io(s); | |
| 91 | + | |
| 92 | + s->cycle = cbus_address; | |
| 93 | + s->bit = 8; | |
| 94 | + s->dir = 1; | |
| 95 | + s->val = 0; | |
| 96 | + break; | |
| 97 | + } | |
| 98 | +} | |
| 99 | + | |
| 100 | +static void cbus_clk(void *opaque, int line, int level) | |
| 101 | +{ | |
| 102 | + struct cbus_priv_s *s = (struct cbus_priv_s *) opaque; | |
| 103 | + | |
| 104 | + if (!s->sel && level && !s->clk) { | |
| 105 | + if (s->dir) | |
| 106 | + s->val |= s->dat << (s->bit --); | |
| 107 | + else | |
| 108 | + qemu_set_irq(s->dat_out, (s->val >> (s->bit --)) & 1); | |
| 109 | + | |
| 110 | + if (s->bit < 0) | |
| 111 | + cbus_cycle(s); | |
| 112 | + } | |
| 113 | + | |
| 114 | + s->clk = level; | |
| 115 | +} | |
| 116 | + | |
| 117 | +static void cbus_dat(void *opaque, int line, int level) | |
| 118 | +{ | |
| 119 | + struct cbus_priv_s *s = (struct cbus_priv_s *) opaque; | |
| 120 | + | |
| 121 | + s->dat = level; | |
| 122 | +} | |
| 123 | + | |
| 124 | +static void cbus_sel(void *opaque, int line, int level) | |
| 125 | +{ | |
| 126 | + struct cbus_priv_s *s = (struct cbus_priv_s *) opaque; | |
| 127 | + | |
| 128 | + if (!level) { | |
| 129 | + s->dir = 1; | |
| 130 | + s->bit = 8; | |
| 131 | + s->val = 0; | |
| 132 | + } | |
| 133 | + | |
| 134 | + s->sel = level; | |
| 135 | +} | |
| 136 | + | |
| 137 | +struct cbus_s *cbus_init(qemu_irq dat) | |
| 138 | +{ | |
| 139 | + struct cbus_priv_s *s = (struct cbus_priv_s *) qemu_mallocz(sizeof(*s)); | |
| 140 | + | |
| 141 | + s->dat_out = dat; | |
| 142 | + s->cbus.clk = qemu_allocate_irqs(cbus_clk, s, 1)[0]; | |
| 143 | + s->cbus.dat = qemu_allocate_irqs(cbus_dat, s, 1)[0]; | |
| 144 | + s->cbus.sel = qemu_allocate_irqs(cbus_sel, s, 1)[0]; | |
| 145 | + | |
| 146 | + s->sel = 1; | |
| 147 | + s->clk = 0; | |
| 148 | + s->dat = 0; | |
| 149 | + | |
| 150 | + return &s->cbus; | |
| 151 | +} | |
| 152 | + | |
| 153 | +void cbus_attach(struct cbus_s *bus, void *slave_opaque) | |
| 154 | +{ | |
| 155 | + struct cbus_slave_s *slave = (struct cbus_slave_s *) slave_opaque; | |
| 156 | + struct cbus_priv_s *s = (struct cbus_priv_s *) bus; | |
| 157 | + | |
| 158 | + s->slave[slave->addr] = slave; | |
| 159 | +} | |
| 160 | + | |
| 161 | +/* Retu/Vilma */ | |
| 162 | +struct cbus_retu_s { | |
| 163 | + uint16_t irqst; | |
| 164 | + uint16_t irqen; | |
| 165 | + uint16_t cc[2]; | |
| 166 | + int channel; | |
| 167 | + uint16_t result[16]; | |
| 168 | + uint16_t sample; | |
| 169 | + uint16_t status; | |
| 170 | + | |
| 171 | + struct { | |
| 172 | + uint16_t cal; | |
| 173 | + } rtc; | |
| 174 | + | |
| 175 | + int is_vilma; | |
| 176 | + qemu_irq irq; | |
| 177 | + struct cbus_slave_s cbus; | |
| 178 | +}; | |
| 179 | + | |
| 180 | +static void retu_interrupt_update(struct cbus_retu_s *s) | |
| 181 | +{ | |
| 182 | + qemu_set_irq(s->irq, s->irqst & ~s->irqen); | |
| 183 | +} | |
| 184 | + | |
| 185 | +#define RETU_REG_ASICR 0x00 /* (RO) ASIC ID & revision */ | |
| 186 | +#define RETU_REG_IDR 0x01 /* (T) Interrupt ID */ | |
| 187 | +#define RETU_REG_IMR 0x02 /* (RW) Interrupt mask */ | |
| 188 | +#define RETU_REG_RTCDSR 0x03 /* (RW) RTC seconds register */ | |
| 189 | +#define RETU_REG_RTCHMR 0x04 /* (RO) RTC hours and minutes reg */ | |
| 190 | +#define RETU_REG_RTCHMAR 0x05 /* (RW) RTC hours and minutes set reg */ | |
| 191 | +#define RETU_REG_RTCCALR 0x06 /* (RW) RTC calibration register */ | |
| 192 | +#define RETU_REG_ADCR 0x08 /* (RW) ADC result register */ | |
| 193 | +#define RETU_REG_ADCSCR 0x09 /* (RW) ADC sample control register */ | |
| 194 | +#define RETU_REG_AFCR 0x0a /* (RW) AFC register */ | |
| 195 | +#define RETU_REG_ANTIFR 0x0b /* (RW) AntiF register */ | |
| 196 | +#define RETU_REG_CALIBR 0x0c /* (RW) CalibR register*/ | |
| 197 | +#define RETU_REG_CCR1 0x0d /* (RW) Common control register 1 */ | |
| 198 | +#define RETU_REG_CCR2 0x0e /* (RW) Common control register 2 */ | |
| 199 | +#define RETU_REG_RCTRL_CLR 0x0f /* (T) Regulator clear register */ | |
| 200 | +#define RETU_REG_RCTRL_SET 0x10 /* (T) Regulator set register */ | |
| 201 | +#define RETU_REG_TXCR 0x11 /* (RW) TxC register */ | |
| 202 | +#define RETU_REG_STATUS 0x16 /* (RO) Status register */ | |
| 203 | +#define RETU_REG_WATCHDOG 0x17 /* (RW) Watchdog register */ | |
| 204 | +#define RETU_REG_AUDTXR 0x18 /* (RW) Audio Codec Tx register */ | |
| 205 | +#define RETU_REG_AUDPAR 0x19 /* (RW) AudioPA register */ | |
| 206 | +#define RETU_REG_AUDRXR1 0x1a /* (RW) Audio receive register 1 */ | |
| 207 | +#define RETU_REG_AUDRXR2 0x1b /* (RW) Audio receive register 2 */ | |
| 208 | +#define RETU_REG_SGR1 0x1c /* (RW) */ | |
| 209 | +#define RETU_REG_SCR1 0x1d /* (RW) */ | |
| 210 | +#define RETU_REG_SGR2 0x1e /* (RW) */ | |
| 211 | +#define RETU_REG_SCR2 0x1f /* (RW) */ | |
| 212 | + | |
| 213 | +/* Retu Interrupt sources */ | |
| 214 | +enum { | |
| 215 | + retu_int_pwr = 0, /* Power button */ | |
| 216 | + retu_int_char = 1, /* Charger */ | |
| 217 | + retu_int_rtcs = 2, /* Seconds */ | |
| 218 | + retu_int_rtcm = 3, /* Minutes */ | |
| 219 | + retu_int_rtcd = 4, /* Days */ | |
| 220 | + retu_int_rtca = 5, /* Alarm */ | |
| 221 | + retu_int_hook = 6, /* Hook */ | |
| 222 | + retu_int_head = 7, /* Headset */ | |
| 223 | + retu_int_adcs = 8, /* ADC sample */ | |
| 224 | +}; | |
| 225 | + | |
| 226 | +/* Retu ADC channel wiring */ | |
| 227 | +enum { | |
| 228 | + retu_adc_bsi = 1, /* BSI */ | |
| 229 | + retu_adc_batt_temp = 2, /* Battery temperature */ | |
| 230 | + retu_adc_chg_volt = 3, /* Charger voltage */ | |
| 231 | + retu_adc_head_det = 4, /* Headset detection */ | |
| 232 | + retu_adc_hook_det = 5, /* Hook detection */ | |
| 233 | + retu_adc_rf_gp = 6, /* RF GP */ | |
| 234 | + retu_adc_tx_det = 7, /* Wideband Tx detection */ | |
| 235 | + retu_adc_batt_volt = 8, /* Battery voltage */ | |
| 236 | + retu_adc_sens = 10, /* Light sensor */ | |
| 237 | + retu_adc_sens_temp = 11, /* Light sensor temperature */ | |
| 238 | + retu_adc_bbatt_volt = 12, /* Backup battery voltage */ | |
| 239 | + retu_adc_self_temp = 13, /* RETU temperature */ | |
| 240 | +}; | |
| 241 | + | |
| 242 | +static inline uint16_t retu_read(struct cbus_retu_s *s, int reg) | |
| 243 | +{ | |
| 244 | +#ifdef DEBUG | |
| 245 | + printf("RETU read at %02x\n", reg); | |
| 246 | +#endif | |
| 247 | + | |
| 248 | + switch (reg) { | |
| 249 | + case RETU_REG_ASICR: | |
| 250 | + return 0x0215 | (s->is_vilma << 7); | |
| 251 | + | |
| 252 | + case RETU_REG_IDR: /* TODO: Or is this ffs(s->irqst)? */ | |
| 253 | + return s->irqst; | |
| 254 | + | |
| 255 | + case RETU_REG_IMR: | |
| 256 | + return s->irqen; | |
| 257 | + | |
| 258 | + case RETU_REG_RTCDSR: | |
| 259 | + case RETU_REG_RTCHMR: | |
| 260 | + case RETU_REG_RTCHMAR: | |
| 261 | + /* TODO */ | |
| 262 | + return 0x0000; | |
| 263 | + | |
| 264 | + case RETU_REG_RTCCALR: | |
| 265 | + return s->rtc.cal; | |
| 266 | + | |
| 267 | + case RETU_REG_ADCR: | |
| 268 | + return (s->channel << 10) | s->result[s->channel]; | |
| 269 | + case RETU_REG_ADCSCR: | |
| 270 | + return s->sample; | |
| 271 | + | |
| 272 | + case RETU_REG_AFCR: | |
| 273 | + case RETU_REG_ANTIFR: | |
| 274 | + case RETU_REG_CALIBR: | |
| 275 | + /* TODO */ | |
| 276 | + return 0x0000; | |
| 277 | + | |
| 278 | + case RETU_REG_CCR1: | |
| 279 | + return s->cc[0]; | |
| 280 | + case RETU_REG_CCR2: | |
| 281 | + return s->cc[1]; | |
| 282 | + | |
| 283 | + case RETU_REG_RCTRL_CLR: | |
| 284 | + case RETU_REG_RCTRL_SET: | |
| 285 | + case RETU_REG_TXCR: | |
| 286 | + /* TODO */ | |
| 287 | + return 0x0000; | |
| 288 | + | |
| 289 | + case RETU_REG_STATUS: | |
| 290 | + return s->status; | |
| 291 | + | |
| 292 | + case RETU_REG_WATCHDOG: | |
| 293 | + case RETU_REG_AUDTXR: | |
| 294 | + case RETU_REG_AUDPAR: | |
| 295 | + case RETU_REG_AUDRXR1: | |
| 296 | + case RETU_REG_AUDRXR2: | |
| 297 | + case RETU_REG_SGR1: | |
| 298 | + case RETU_REG_SCR1: | |
| 299 | + case RETU_REG_SGR2: | |
| 300 | + case RETU_REG_SCR2: | |
| 301 | + /* TODO */ | |
| 302 | + return 0x0000; | |
| 303 | + | |
| 304 | + default: | |
| 305 | + cpu_abort(cpu_single_env, "%s: bad register %02x\n", | |
| 306 | + __FUNCTION__, reg); | |
| 307 | + } | |
| 308 | +} | |
| 309 | + | |
| 310 | +static inline void retu_write(struct cbus_retu_s *s, int reg, uint16_t val) | |
| 311 | +{ | |
| 312 | +#ifdef DEBUG | |
| 313 | + printf("RETU write of %04x at %02x\n", val, reg); | |
| 314 | +#endif | |
| 315 | + | |
| 316 | + switch (reg) { | |
| 317 | + case RETU_REG_IDR: | |
| 318 | + s->irqst ^= val; | |
| 319 | + retu_interrupt_update(s); | |
| 320 | + break; | |
| 321 | + | |
| 322 | + case RETU_REG_IMR: | |
| 323 | + s->irqen = val; | |
| 324 | + retu_interrupt_update(s); | |
| 325 | + break; | |
| 326 | + | |
| 327 | + case RETU_REG_RTCDSR: | |
| 328 | + case RETU_REG_RTCHMAR: | |
| 329 | + /* TODO */ | |
| 330 | + break; | |
| 331 | + | |
| 332 | + case RETU_REG_RTCCALR: | |
| 333 | + s->rtc.cal = val; | |
| 334 | + break; | |
| 335 | + | |
| 336 | + case RETU_REG_ADCR: | |
| 337 | + s->channel = (val >> 10) & 0xf; | |
| 338 | + s->irqst |= 1 << retu_int_adcs; | |
| 339 | + retu_interrupt_update(s); | |
| 340 | + break; | |
| 341 | + case RETU_REG_ADCSCR: | |
| 342 | + s->sample &= ~val; | |
| 343 | + break; | |
| 344 | + | |
| 345 | + case RETU_REG_AFCR: | |
| 346 | + case RETU_REG_ANTIFR: | |
| 347 | + case RETU_REG_CALIBR: | |
| 348 | + | |
| 349 | + case RETU_REG_CCR1: | |
| 350 | + s->cc[0] = val; | |
| 351 | + break; | |
| 352 | + case RETU_REG_CCR2: | |
| 353 | + s->cc[1] = val; | |
| 354 | + break; | |
| 355 | + | |
| 356 | + case RETU_REG_RCTRL_CLR: | |
| 357 | + case RETU_REG_RCTRL_SET: | |
| 358 | + /* TODO */ | |
| 359 | + break; | |
| 360 | + | |
| 361 | + case RETU_REG_WATCHDOG: | |
| 362 | + if (val == 0 && (s->cc[0] & 2)) | |
| 363 | + qemu_system_shutdown_request(); | |
| 364 | + break; | |
| 365 | + | |
| 366 | + case RETU_REG_TXCR: | |
| 367 | + case RETU_REG_AUDTXR: | |
| 368 | + case RETU_REG_AUDPAR: | |
| 369 | + case RETU_REG_AUDRXR1: | |
| 370 | + case RETU_REG_AUDRXR2: | |
| 371 | + case RETU_REG_SGR1: | |
| 372 | + case RETU_REG_SCR1: | |
| 373 | + case RETU_REG_SGR2: | |
| 374 | + case RETU_REG_SCR2: | |
| 375 | + /* TODO */ | |
| 376 | + break; | |
| 377 | + | |
| 378 | + default: | |
| 379 | + cpu_abort(cpu_single_env, "%s: bad register %02x\n", | |
| 380 | + __FUNCTION__, reg); | |
| 381 | + } | |
| 382 | +} | |
| 383 | + | |
| 384 | +static void retu_io(void *opaque, int rw, int reg, uint16_t *val) | |
| 385 | +{ | |
| 386 | + struct cbus_retu_s *s = (struct cbus_retu_s *) opaque; | |
| 387 | + | |
| 388 | + if (rw) | |
| 389 | + *val = retu_read(s, reg); | |
| 390 | + else | |
| 391 | + retu_write(s, reg, *val); | |
| 392 | +} | |
| 393 | + | |
| 394 | +void *retu_init(qemu_irq irq, int vilma) | |
| 395 | +{ | |
| 396 | + struct cbus_retu_s *s = (struct cbus_retu_s *) qemu_mallocz(sizeof(*s)); | |
| 397 | + | |
| 398 | + s->irq = irq; | |
| 399 | + s->irqen = 0xffff; | |
| 400 | + s->irqst = 0x0000; | |
| 401 | + s->status = 0x0020; | |
| 402 | + s->is_vilma = !!vilma; | |
| 403 | + s->rtc.cal = 0x01; | |
| 404 | + s->result[retu_adc_bsi] = 0x3c2; | |
| 405 | + s->result[retu_adc_batt_temp] = 0x0fc; | |
| 406 | + s->result[retu_adc_chg_volt] = 0x165; | |
| 407 | + s->result[retu_adc_head_det] = 123; | |
| 408 | + s->result[retu_adc_hook_det] = 1023; | |
| 409 | + s->result[retu_adc_rf_gp] = 0x11; | |
| 410 | + s->result[retu_adc_tx_det] = 0x11; | |
| 411 | + s->result[retu_adc_batt_volt] = 0x250; | |
| 412 | + s->result[retu_adc_sens] = 2; | |
| 413 | + s->result[retu_adc_sens_temp] = 0x11; | |
| 414 | + s->result[retu_adc_bbatt_volt] = 0x3d0; | |
| 415 | + s->result[retu_adc_self_temp] = 0x330; | |
| 416 | + | |
| 417 | + s->cbus.opaque = s; | |
| 418 | + s->cbus.io = retu_io; | |
| 419 | + s->cbus.addr = 1; | |
| 420 | + | |
| 421 | + return &s->cbus; | |
| 422 | +} | |
| 423 | + | |
| 424 | +void retu_key_event(void *retu, int state) | |
| 425 | +{ | |
| 426 | + struct cbus_slave_s *slave = (struct cbus_slave_s *) retu; | |
| 427 | + struct cbus_retu_s *s = (struct cbus_retu_s *) slave->opaque; | |
| 428 | + | |
| 429 | + s->irqst |= 1 << retu_int_pwr; | |
| 430 | + retu_interrupt_update(s); | |
| 431 | + | |
| 432 | + if (state) | |
| 433 | + s->status &= ~(1 << 5); | |
| 434 | + else | |
| 435 | + s->status |= 1 << 5; | |
| 436 | +} | |
| 437 | + | |
| 438 | +void retu_head_event(void *retu, int state) | |
| 439 | +{ | |
| 440 | + struct cbus_slave_s *slave = (struct cbus_slave_s *) retu; | |
| 441 | + struct cbus_retu_s *s = (struct cbus_retu_s *) slave->opaque; | |
| 442 | + | |
| 443 | + if ((s->cc[0] & 0x500) == 0x500) { /* TODO: Which bits? */ | |
| 444 | + /* TODO: reissue the interrupt every 100ms or so. */ | |
| 445 | + s->irqst |= 1 << retu_int_head; | |
| 446 | + retu_interrupt_update(s); | |
| 447 | + } | |
| 448 | + | |
| 449 | + if (state) | |
| 450 | + s->result[retu_adc_head_det] = 50; | |
| 451 | + else | |
| 452 | + s->result[retu_adc_head_det] = 123; | |
| 453 | +} | |
| 454 | + | |
| 455 | +void retu_hook_event(void *retu, int state) | |
| 456 | +{ | |
| 457 | + struct cbus_slave_s *slave = (struct cbus_slave_s *) retu; | |
| 458 | + struct cbus_retu_s *s = (struct cbus_retu_s *) slave->opaque; | |
| 459 | + | |
| 460 | + if ((s->cc[0] & 0x500) == 0x500) { | |
| 461 | + /* TODO: reissue the interrupt every 100ms or so. */ | |
| 462 | + s->irqst |= 1 << retu_int_hook; | |
| 463 | + retu_interrupt_update(s); | |
| 464 | + } | |
| 465 | + | |
| 466 | + if (state) | |
| 467 | + s->result[retu_adc_hook_det] = 50; | |
| 468 | + else | |
| 469 | + s->result[retu_adc_hook_det] = 123; | |
| 470 | +} | |
| 471 | + | |
| 472 | +/* Tahvo/Betty */ | |
| 473 | +struct cbus_tahvo_s { | |
| 474 | + uint16_t irqst; | |
| 475 | + uint16_t irqen; | |
| 476 | + uint8_t charger; | |
| 477 | + uint8_t backlight; | |
| 478 | + uint16_t usbr; | |
| 479 | + uint16_t power; | |
| 480 | + | |
| 481 | + int is_betty; | |
| 482 | + qemu_irq irq; | |
| 483 | + struct cbus_slave_s cbus; | |
| 484 | +}; | |
| 485 | + | |
| 486 | +static void tahvo_interrupt_update(struct cbus_tahvo_s *s) | |
| 487 | +{ | |
| 488 | + qemu_set_irq(s->irq, s->irqst & ~s->irqen); | |
| 489 | +} | |
| 490 | + | |
| 491 | +#define TAHVO_REG_ASICR 0x00 /* (RO) ASIC ID & revision */ | |
| 492 | +#define TAHVO_REG_IDR 0x01 /* (T) Interrupt ID */ | |
| 493 | +#define TAHVO_REG_IDSR 0x02 /* (RO) Interrupt status */ | |
| 494 | +#define TAHVO_REG_IMR 0x03 /* (RW) Interrupt mask */ | |
| 495 | +#define TAHVO_REG_CHAPWMR 0x04 /* (RW) Charger PWM */ | |
| 496 | +#define TAHVO_REG_LEDPWMR 0x05 /* (RW) LED PWM */ | |
| 497 | +#define TAHVO_REG_USBR 0x06 /* (RW) USB control */ | |
| 498 | +#define TAHVO_REG_RCR 0x07 /* (RW) Some kind of power management */ | |
| 499 | +#define TAHVO_REG_CCR1 0x08 /* (RW) Common control register 1 */ | |
| 500 | +#define TAHVO_REG_CCR2 0x09 /* (RW) Common control register 2 */ | |
| 501 | +#define TAHVO_REG_TESTR1 0x0a /* (RW) Test register 1 */ | |
| 502 | +#define TAHVO_REG_TESTR2 0x0b /* (RW) Test register 2 */ | |
| 503 | +#define TAHVO_REG_NOPR 0x0c /* (RW) Number of periods */ | |
| 504 | +#define TAHVO_REG_FRR 0x0d /* (RO) FR */ | |
| 505 | + | |
| 506 | +static inline uint16_t tahvo_read(struct cbus_tahvo_s *s, int reg) | |
| 507 | +{ | |
| 508 | +#ifdef DEBUG | |
| 509 | + printf("TAHVO read at %02x\n", reg); | |
| 510 | +#endif | |
| 511 | + | |
| 512 | + switch (reg) { | |
| 513 | + case TAHVO_REG_ASICR: | |
| 514 | + return 0x0021 | (s->is_betty ? 0x0b00 : 0x0300); /* 22 in N810 */ | |
| 515 | + | |
| 516 | + case TAHVO_REG_IDR: | |
| 517 | + case TAHVO_REG_IDSR: /* XXX: what does this do? */ | |
| 518 | + return s->irqst; | |
| 519 | + | |
| 520 | + case TAHVO_REG_IMR: | |
| 521 | + return s->irqen; | |
| 522 | + | |
| 523 | + case TAHVO_REG_CHAPWMR: | |
| 524 | + return s->charger; | |
| 525 | + | |
| 526 | + case TAHVO_REG_LEDPWMR: | |
| 527 | + return s->backlight; | |
| 528 | + | |
| 529 | + case TAHVO_REG_USBR: | |
| 530 | + return s->usbr; | |
| 531 | + | |
| 532 | + case TAHVO_REG_RCR: | |
| 533 | + return s->power; | |
| 534 | + | |
| 535 | + case TAHVO_REG_CCR1: | |
| 536 | + case TAHVO_REG_CCR2: | |
| 537 | + case TAHVO_REG_TESTR1: | |
| 538 | + case TAHVO_REG_TESTR2: | |
| 539 | + case TAHVO_REG_NOPR: | |
| 540 | + case TAHVO_REG_FRR: | |
| 541 | + return 0x0000; | |
| 542 | + | |
| 543 | + default: | |
| 544 | + cpu_abort(cpu_single_env, "%s: bad register %02x\n", | |
| 545 | + __FUNCTION__, reg); | |
| 546 | + } | |
| 547 | +} | |
| 548 | + | |
| 549 | +static inline void tahvo_write(struct cbus_tahvo_s *s, int reg, uint16_t val) | |
| 550 | +{ | |
| 551 | +#ifdef DEBUG | |
| 552 | + printf("TAHVO write of %04x at %02x\n", val, reg); | |
| 553 | +#endif | |
| 554 | + | |
| 555 | + switch (reg) { | |
| 556 | + case TAHVO_REG_IDR: | |
| 557 | + s->irqst ^= val; | |
| 558 | + tahvo_interrupt_update(s); | |
| 559 | + break; | |
| 560 | + | |
| 561 | + case TAHVO_REG_IMR: | |
| 562 | + s->irqen = val; | |
| 563 | + tahvo_interrupt_update(s); | |
| 564 | + break; | |
| 565 | + | |
| 566 | + case TAHVO_REG_CHAPWMR: | |
| 567 | + s->charger = val; | |
| 568 | + break; | |
| 569 | + | |
| 570 | + case TAHVO_REG_LEDPWMR: | |
| 571 | + if (s->backlight != (val & 0x7f)) { | |
| 572 | + s->backlight = val & 0x7f; | |
| 573 | + printf("%s: LCD backlight now at %i / 127\n", | |
| 574 | + __FUNCTION__, s->backlight); | |
| 575 | + } | |
| 576 | + break; | |
| 577 | + | |
| 578 | + case TAHVO_REG_USBR: | |
| 579 | + s->usbr = val; | |
| 580 | + break; | |
| 581 | + | |
| 582 | + case TAHVO_REG_RCR: | |
| 583 | + s->power = val; | |
| 584 | + break; | |
| 585 | + | |
| 586 | + case TAHVO_REG_CCR1: | |
| 587 | + case TAHVO_REG_CCR2: | |
| 588 | + case TAHVO_REG_TESTR1: | |
| 589 | + case TAHVO_REG_TESTR2: | |
| 590 | + case TAHVO_REG_NOPR: | |
| 591 | + case TAHVO_REG_FRR: | |
| 592 | + break; | |
| 593 | + | |
| 594 | + default: | |
| 595 | + cpu_abort(cpu_single_env, "%s: bad register %02x\n", | |
| 596 | + __FUNCTION__, reg); | |
| 597 | + } | |
| 598 | +} | |
| 599 | + | |
| 600 | +static void tahvo_io(void *opaque, int rw, int reg, uint16_t *val) | |
| 601 | +{ | |
| 602 | + struct cbus_tahvo_s *s = (struct cbus_tahvo_s *) opaque; | |
| 603 | + | |
| 604 | + if (rw) | |
| 605 | + *val = tahvo_read(s, reg); | |
| 606 | + else | |
| 607 | + tahvo_write(s, reg, *val); | |
| 608 | +} | |
| 609 | + | |
| 610 | +void *tahvo_init(qemu_irq irq, int betty) | |
| 611 | +{ | |
| 612 | + struct cbus_tahvo_s *s = (struct cbus_tahvo_s *) qemu_mallocz(sizeof(*s)); | |
| 613 | + | |
| 614 | + s->irq = irq; | |
| 615 | + s->irqen = 0xffff; | |
| 616 | + s->irqst = 0x0000; | |
| 617 | + s->is_betty = !!betty; | |
| 618 | + | |
| 619 | + s->cbus.opaque = s; | |
| 620 | + s->cbus.io = tahvo_io; | |
| 621 | + s->cbus.addr = 2; | |
| 622 | + | |
| 623 | + return &s->cbus; | |
| 624 | +} | ... | ... |
hw/devices.h
| ... | ... | @@ -31,4 +31,25 @@ void tsc210x_key_event(struct uwire_slave_s *chip, int key, int down); |
| 31 | 31 | /* stellaris_input.c */ |
| 32 | 32 | void stellaris_gamepad_init(int n, qemu_irq *irq, const int *keycode); |
| 33 | 33 | |
| 34 | +/* blizzard.c */ | |
| 35 | +void *s1d13745_init(qemu_irq gpio_int, DisplayState *ds); | |
| 36 | +void s1d13745_write(void *opaque, int dc, uint16_t value); | |
| 37 | +void s1d13745_write_block(void *opaque, int dc, | |
| 38 | + void *buf, size_t len, int pitch); | |
| 39 | +uint16_t s1d13745_read(void *opaque, int dc); | |
| 40 | + | |
| 41 | +/* cbus.c */ | |
| 42 | +struct cbus_s { | |
| 43 | + qemu_irq clk; | |
| 44 | + qemu_irq dat; | |
| 45 | + qemu_irq sel; | |
| 46 | +}; | |
| 47 | +struct cbus_s *cbus_init(qemu_irq dat_out); | |
| 48 | +void cbus_attach(struct cbus_s *bus, void *slave_opaque); | |
| 49 | + | |
| 50 | +void *retu_init(qemu_irq irq, int vilma); | |
| 51 | +void *tahvo_init(qemu_irq irq, int betty); | |
| 52 | + | |
| 53 | +void retu_key_event(void *retu, int state); | |
| 54 | + | |
| 34 | 55 | #endif | ... | ... |
hw/flash.h
| ... | ... | @@ -34,6 +34,11 @@ uint8_t nand_getio(struct nand_flash_s *s); |
| 34 | 34 | #define NAND_MFR_HYNIX 0xad |
| 35 | 35 | #define NAND_MFR_MICRON 0x2c |
| 36 | 36 | |
| 37 | +/* onenand.c */ | |
| 38 | +void onenand_base_update(void *opaque, target_phys_addr_t new); | |
| 39 | +void onenand_base_unmap(void *opaque); | |
| 40 | +void *onenand_init(uint32_t id, int regshift, qemu_irq irq); | |
| 41 | + | |
| 37 | 42 | /* ecc.c */ |
| 38 | 43 | struct ecc_state_s { |
| 39 | 44 | uint8_t cp; /* Column parity */ | ... | ... |
hw/i2c.h
| ... | ... | @@ -71,4 +71,14 @@ uint32_t wm8750_adc_dat(void *opaque); |
| 71 | 71 | /* ssd0303.c */ |
| 72 | 72 | void ssd0303_init(DisplayState *ds, i2c_bus *bus, int address); |
| 73 | 73 | |
| 74 | +/* twl92230.c */ | |
| 75 | +i2c_slave *twl92230_init(i2c_bus *bus, qemu_irq irq); | |
| 76 | +qemu_irq *twl92230_gpio_in_get(i2c_slave *i2c); | |
| 77 | +void twl92230_gpio_out_set(i2c_slave *i2c, int line, qemu_irq handler); | |
| 78 | + | |
| 79 | +/* tmp105.c */ | |
| 80 | +struct i2c_slave *tmp105_init(i2c_bus *bus, qemu_irq alarm); | |
| 81 | +void tmp105_reset(i2c_slave *i2c); | |
| 82 | +void tmp105_set(i2c_slave *i2c, int temp); | |
| 83 | + | |
| 74 | 84 | #endif | ... | ... |
hw/nseries.c
0 → 100644
| 1 | +/* | |
| 2 | + * Nokia N-series internet tablets. | |
| 3 | + * | |
| 4 | + * Copyright (C) 2007 Nokia Corporation | |
| 5 | + * Written by Andrzej Zaborowski <andrew@openedhand.com> | |
| 6 | + * | |
| 7 | + * This program is free software; you can redistribute it and/or | |
| 8 | + * modify it under the terms of the GNU General Public License as | |
| 9 | + * published by the Free Software Foundation; either version 2 or | |
| 10 | + * (at your option) version 3 of the License. | |
| 11 | + * | |
| 12 | + * This program is distributed in the hope that it will be useful, | |
| 13 | + * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
| 14 | + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
| 15 | + * GNU General Public License for more details. | |
| 16 | + * | |
| 17 | + * You should have received a copy of the GNU General Public License | |
| 18 | + * along with this program; if not, write to the Free Software | |
| 19 | + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, | |
| 20 | + * MA 02111-1307 USA | |
| 21 | + */ | |
| 22 | + | |
| 23 | +#include "qemu-common.h" | |
| 24 | +#include "sysemu.h" | |
| 25 | +#include "omap.h" | |
| 26 | +#include "arm-misc.h" | |
| 27 | +#include "irq.h" | |
| 28 | +#include "console.h" | |
| 29 | +#include "boards.h" | |
| 30 | +#include "i2c.h" | |
| 31 | +#include "devices.h" | |
| 32 | +#include "flash.h" | |
| 33 | +#include "hw.h" | |
| 34 | + | |
| 35 | +/* Nokia N8x0 support */ | |
| 36 | +struct n800_s { | |
| 37 | + struct omap_mpu_state_s *cpu; | |
| 38 | + | |
| 39 | + struct rfbi_chip_s blizzard; | |
| 40 | + struct uwire_slave_s *ts; | |
| 41 | + i2c_bus *i2c; | |
| 42 | + | |
| 43 | + int keymap[0x80]; | |
| 44 | + | |
| 45 | + void *retu; | |
| 46 | + void *tahvo; | |
| 47 | +}; | |
| 48 | + | |
| 49 | +/* GPIO pins */ | |
| 50 | +#define N800_TUSB_ENABLE_GPIO 0 | |
| 51 | +#define N800_MMC2_WP_GPIO 8 | |
| 52 | +#define N800_UNKNOWN_GPIO0 9 /* out */ | |
| 53 | +#define N800_UNKNOWN_GPIO1 10 /* out */ | |
| 54 | +#define N800_CAM_TURN_GPIO 12 | |
| 55 | +#define N800_BLIZZARD_POWERDOWN_GPIO 15 | |
| 56 | +#define N800_MMC1_WP_GPIO 23 | |
| 57 | +#define N8X0_ONENAND_GPIO 26 | |
| 58 | +#define N800_UNKNOWN_GPIO2 53 /* out */ | |
| 59 | +#define N8X0_TUSB_INT_GPIO 58 | |
| 60 | +#define N800_BT_WKUP_GPIO 61 | |
| 61 | +#define N800_STI_GPIO 62 | |
| 62 | +#define N8X0_CBUS_SEL_GPIO 64 | |
| 63 | +#define N8X0_CBUS_CLK_GPIO 65 /* sure? */ | |
| 64 | +#define N8X0_CBUS_DAT_GPIO 66 | |
| 65 | +#define N800_WLAN_IRQ_GPIO 87 | |
| 66 | +#define N800_BT_RESET_GPIO 92 | |
| 67 | +#define N800_TEA5761_CS_GPIO 93 | |
| 68 | +#define N800_UNKNOWN_GPIO 94 | |
| 69 | +#define N800_CAM_ACT_GPIO 95 | |
| 70 | +#define N800_MMC_CS_GPIO 96 | |
| 71 | +#define N800_WLAN_PWR_GPIO 97 | |
| 72 | +#define N8X0_BT_HOST_WKUP_GPIO 98 | |
| 73 | +#define N800_UNKNOWN_GPIO3 101 /* out */ | |
| 74 | +#define N810_KB_LOCK_GPIO 102 | |
| 75 | +#define N800_TSC_TS_GPIO 103 | |
| 76 | +#define N810_TSC2005_GPIO 106 | |
| 77 | +#define N800_HEADPHONE_GPIO 107 | |
| 78 | +#define N8X0_RETU_GPIO 108 | |
| 79 | +#define N800_TSC_KP_IRQ_GPIO 109 | |
| 80 | +#define N810_KEYBOARD_GPIO 109 | |
| 81 | +#define N800_BAT_COVER_GPIO 110 | |
| 82 | +#define N810_SLIDE_GPIO 110 | |
| 83 | +#define N8X0_TAHVO_GPIO 111 | |
| 84 | +#define N800_UNKNOWN_GPIO4 112 /* out */ | |
| 85 | +#define N810_TSC_RESET_GPIO 118 | |
| 86 | +#define N800_TSC_RESET_GPIO 119 /* ? */ | |
| 87 | +#define N8X0_TMP105_GPIO 125 | |
| 88 | + | |
| 89 | +/* Config */ | |
| 90 | +#define XLDR_LL_UART 1 | |
| 91 | + | |
| 92 | +/* Addresses on the I2C bus */ | |
| 93 | +#define N8X0_TMP105_ADDR 0x48 | |
| 94 | +#define N8X0_MENELAUS_ADDR 0x72 | |
| 95 | + | |
| 96 | +/* Chipselects on GPMC NOR interface */ | |
| 97 | +#define N8X0_ONENAND_CS 0 | |
| 98 | +#define N8X0_USB_ASYNC_CS 1 | |
| 99 | +#define N8X0_USB_SYNC_CS 4 | |
| 100 | + | |
| 101 | +static void n800_mmc_cs_cb(void *opaque, int line, int level) | |
| 102 | +{ | |
| 103 | + /* TODO: this seems to actually be connected to the menelaus, to | |
| 104 | + * which also both MMC slots connect. */ | |
| 105 | + omap_mmc_enable((struct omap_mmc_s *) opaque, !level); | |
| 106 | + | |
| 107 | + printf("%s: MMC slot %i active\n", __FUNCTION__, level + 1); | |
| 108 | +} | |
| 109 | + | |
| 110 | +static void n800_gpio_setup(struct n800_s *s) | |
| 111 | +{ | |
| 112 | + qemu_irq *mmc_cs = qemu_allocate_irqs(n800_mmc_cs_cb, s->cpu->mmc, 1); | |
| 113 | + omap2_gpio_out_set(s->cpu->gpif, N800_MMC_CS_GPIO, mmc_cs[0]); | |
| 114 | + | |
| 115 | + qemu_irq_lower(omap2_gpio_in_get(s->cpu->gpif, N800_BAT_COVER_GPIO)[0]); | |
| 116 | +} | |
| 117 | + | |
| 118 | +static void n8x0_nand_setup(struct n800_s *s) | |
| 119 | +{ | |
| 120 | + /* Either ec40xx or ec48xx are OK for the ID */ | |
| 121 | + omap_gpmc_attach(s->cpu->gpmc, N8X0_ONENAND_CS, 0, onenand_base_update, | |
| 122 | + onenand_base_unmap, | |
| 123 | + onenand_init(0xec4800, 1, | |
| 124 | + omap2_gpio_in_get(s->cpu->gpif, | |
| 125 | + N8X0_ONENAND_GPIO)[0])); | |
| 126 | +} | |
| 127 | + | |
| 128 | +static void n800_i2c_setup(struct n800_s *s) | |
| 129 | +{ | |
| 130 | + qemu_irq tmp_irq = omap2_gpio_in_get(s->cpu->gpif, N8X0_TMP105_GPIO)[0]; | |
| 131 | + | |
| 132 | + /* Attach the CPU on one end of our I2C bus. */ | |
| 133 | + s->i2c = omap_i2c_bus(s->cpu->i2c[0]); | |
| 134 | + | |
| 135 | + /* Attach a menelaus PM chip */ | |
| 136 | + i2c_set_slave_address( | |
| 137 | + twl92230_init(s->i2c, | |
| 138 | + s->cpu->irq[0][OMAP_INT_24XX_SYS_NIRQ]), | |
| 139 | + N8X0_MENELAUS_ADDR); | |
| 140 | + | |
| 141 | + /* Attach a TMP105 PM chip (A0 wired to ground) */ | |
| 142 | + i2c_set_slave_address(tmp105_init(s->i2c, tmp_irq), N8X0_TMP105_ADDR); | |
| 143 | +} | |
| 144 | + | |
| 145 | +/* Touchscreen and keypad controller */ | |
| 146 | +#define RETU_KEYCODE 61 /* F3 */ | |
| 147 | + | |
| 148 | +static void n800_key_event(void *opaque, int keycode) | |
| 149 | +{ | |
| 150 | + struct n800_s *s = (struct n800_s *) opaque; | |
| 151 | + int code = s->keymap[keycode & 0x7f]; | |
| 152 | + | |
| 153 | + if (code == -1) { | |
| 154 | + if ((keycode & 0x7f) == RETU_KEYCODE) | |
| 155 | + retu_key_event(s->retu, !(keycode & 0x80)); | |
| 156 | + return; | |
| 157 | + } | |
| 158 | + | |
| 159 | + tsc210x_key_event(s->ts, code, !(keycode & 0x80)); | |
| 160 | +} | |
| 161 | + | |
| 162 | +static const int n800_keys[16] = { | |
| 163 | + -1, | |
| 164 | + 72, /* Up */ | |
| 165 | + 63, /* Home (F5) */ | |
| 166 | + -1, | |
| 167 | + 75, /* Left */ | |
| 168 | + 28, /* Enter */ | |
| 169 | + 77, /* Right */ | |
| 170 | + -1, | |
| 171 | + 1, /* Cycle (ESC) */ | |
| 172 | + 80, /* Down */ | |
| 173 | + 62, /* Menu (F4) */ | |
| 174 | + -1, | |
| 175 | + 66, /* Zoom- (F8) */ | |
| 176 | + 64, /* FS (F6) */ | |
| 177 | + 65, /* Zoom+ (F7) */ | |
| 178 | + -1, | |
| 179 | +}; | |
| 180 | + | |
| 181 | +static struct mouse_transform_info_s n800_pointercal = { | |
| 182 | + .x = 800, | |
| 183 | + .y = 480, | |
| 184 | + .a = { 14560, -68, -3455208, -39, -9621, 35152972, 65536 }, | |
| 185 | +}; | |
| 186 | + | |
| 187 | +static void n800_tsc_setup(struct n800_s *s) | |
| 188 | +{ | |
| 189 | + int i; | |
| 190 | + | |
| 191 | + /* XXX: are the three pins inverted inside the chip between the | |
| 192 | + * tsc and the cpu (N4111)? */ | |
| 193 | + qemu_irq penirq = 0; /* NC */ | |
| 194 | + qemu_irq kbirq = omap2_gpio_in_get(s->cpu->gpif, N800_TSC_KP_IRQ_GPIO)[0]; | |
| 195 | + qemu_irq dav = omap2_gpio_in_get(s->cpu->gpif, N800_TSC_TS_GPIO)[0]; | |
| 196 | + | |
| 197 | + s->ts = tsc2301_init(penirq, kbirq, dav, 0); | |
| 198 | + | |
| 199 | + for (i = 0; i < 0x80; i ++) | |
| 200 | + s->keymap[i] = -1; | |
| 201 | + for (i = 0; i < 0x10; i ++) | |
| 202 | + if (n800_keys[i] >= 0) | |
| 203 | + s->keymap[n800_keys[i]] = i; | |
| 204 | + | |
| 205 | + qemu_add_kbd_event_handler(n800_key_event, s); | |
| 206 | + | |
| 207 | + tsc210x_set_transform(s->ts, &n800_pointercal); | |
| 208 | +} | |
| 209 | + | |
| 210 | +/* LCD MIPI DBI-C controller (URAL) */ | |
| 211 | +struct mipid_s { | |
| 212 | + int resp[4]; | |
| 213 | + int param[4]; | |
| 214 | + int p; | |
| 215 | + int pm; | |
| 216 | + int cmd; | |
| 217 | + | |
| 218 | + int sleep; | |
| 219 | + int booster; | |
| 220 | + int te; | |
| 221 | + int selfcheck; | |
| 222 | + int partial; | |
| 223 | + int normal; | |
| 224 | + int vscr; | |
| 225 | + int invert; | |
| 226 | + int onoff; | |
| 227 | + int gamma; | |
| 228 | + uint32_t id; | |
| 229 | +}; | |
| 230 | + | |
| 231 | +static void mipid_reset(struct mipid_s *s) | |
| 232 | +{ | |
| 233 | + if (!s->sleep) | |
| 234 | + fprintf(stderr, "%s: Display off\n", __FUNCTION__); | |
| 235 | + | |
| 236 | + s->pm = 0; | |
| 237 | + s->cmd = 0; | |
| 238 | + | |
| 239 | + s->sleep = 1; | |
| 240 | + s->booster = 0; | |
| 241 | + s->selfcheck = | |
| 242 | + (1 << 7) | /* Register loading OK. */ | |
| 243 | + (1 << 5) | /* The chip is attached. */ | |
| 244 | + (1 << 4); /* Display glass still in one piece. */ | |
| 245 | + s->te = 0; | |
| 246 | + s->partial = 0; | |
| 247 | + s->normal = 1; | |
| 248 | + s->vscr = 0; | |
| 249 | + s->invert = 0; | |
| 250 | + s->onoff = 1; | |
| 251 | + s->gamma = 0; | |
| 252 | +} | |
| 253 | + | |
| 254 | +static uint32_t mipid_txrx(void *opaque, uint32_t cmd) | |
| 255 | +{ | |
| 256 | + struct mipid_s *s = (struct mipid_s *) opaque; | |
| 257 | + uint8_t ret; | |
| 258 | + | |
| 259 | + if (s->p >= sizeof(s->resp) / sizeof(*s->resp)) | |
| 260 | + ret = 0; | |
| 261 | + else | |
| 262 | + ret = s->resp[s->p ++]; | |
| 263 | + if (s->pm --> 0) | |
| 264 | + s->param[s->pm] = cmd; | |
| 265 | + else | |
| 266 | + s->cmd = cmd; | |
| 267 | + | |
| 268 | + switch (s->cmd) { | |
| 269 | + case 0x00: /* NOP */ | |
| 270 | + break; | |
| 271 | + | |
| 272 | + case 0x01: /* SWRESET */ | |
| 273 | + mipid_reset(s); | |
| 274 | + break; | |
| 275 | + | |
| 276 | + case 0x02: /* BSTROFF */ | |
| 277 | + s->booster = 0; | |
| 278 | + break; | |
| 279 | + case 0x03: /* BSTRON */ | |
| 280 | + s->booster = 1; | |
| 281 | + break; | |
| 282 | + | |
| 283 | + case 0x04: /* RDDID */ | |
| 284 | + s->p = 0; | |
| 285 | + s->resp[0] = (s->id >> 16) & 0xff; | |
| 286 | + s->resp[1] = (s->id >> 8) & 0xff; | |
| 287 | + s->resp[2] = (s->id >> 0) & 0xff; | |
| 288 | + break; | |
| 289 | + | |
| 290 | + case 0x06: /* RD_RED */ | |
| 291 | + case 0x07: /* RD_GREEN */ | |
| 292 | + /* XXX the bootloader sometimes issues RD_BLUE meaning RDDID so | |
| 293 | + * for the bootloader one needs to change this. */ | |
| 294 | + case 0x08: /* RD_BLUE */ | |
| 295 | + s->p = 0; | |
| 296 | + /* TODO: return first pixel components */ | |
| 297 | + s->resp[0] = 0x01; | |
| 298 | + break; | |
| 299 | + | |
| 300 | + case 0x09: /* RDDST */ | |
| 301 | + s->p = 0; | |
| 302 | + s->resp[0] = s->booster << 7; | |
| 303 | + s->resp[1] = (5 << 4) | (s->partial << 2) | | |
| 304 | + (s->sleep << 1) | s->normal; | |
| 305 | + s->resp[2] = (s->vscr << 7) | (s->invert << 5) | | |
| 306 | + (s->onoff << 2) | (s->te << 1) | (s->gamma >> 2); | |
| 307 | + s->resp[3] = s->gamma << 6; | |
| 308 | + break; | |
| 309 | + | |
| 310 | + case 0x0a: /* RDDPM */ | |
| 311 | + s->p = 0; | |
| 312 | + s->resp[0] = (s->onoff << 2) | (s->normal << 3) | (s->sleep << 4) | | |
| 313 | + (s->partial << 5) | (s->sleep << 6) | (s->booster << 7); | |
| 314 | + break; | |
| 315 | + case 0x0b: /* RDDMADCTR */ | |
| 316 | + s->p = 0; | |
| 317 | + s->resp[0] = 0; | |
| 318 | + break; | |
| 319 | + case 0x0c: /* RDDCOLMOD */ | |
| 320 | + s->p = 0; | |
| 321 | + s->resp[0] = 5; /* 65K colours */ | |
| 322 | + break; | |
| 323 | + case 0x0d: /* RDDIM */ | |
| 324 | + s->p = 0; | |
| 325 | + s->resp[0] = (s->invert << 5) | (s->vscr << 7) | s->gamma; | |
| 326 | + break; | |
| 327 | + case 0x0e: /* RDDSM */ | |
| 328 | + s->p = 0; | |
| 329 | + s->resp[0] = s->te << 7; | |
| 330 | + break; | |
| 331 | + case 0x0f: /* RDDSDR */ | |
| 332 | + s->p = 0; | |
| 333 | + s->resp[0] = s->selfcheck; | |
| 334 | + break; | |
| 335 | + | |
| 336 | + case 0x10: /* SLPIN */ | |
| 337 | + s->sleep = 1; | |
| 338 | + break; | |
| 339 | + case 0x11: /* SLPOUT */ | |
| 340 | + s->sleep = 0; | |
| 341 | + s->selfcheck ^= 1 << 6; /* POFF self-diagnosis Ok */ | |
| 342 | + break; | |
| 343 | + | |
| 344 | + case 0x12: /* PTLON */ | |
| 345 | + s->partial = 1; | |
| 346 | + s->normal = 0; | |
| 347 | + s->vscr = 0; | |
| 348 | + break; | |
| 349 | + case 0x13: /* NORON */ | |
| 350 | + s->partial = 0; | |
| 351 | + s->normal = 1; | |
| 352 | + s->vscr = 0; | |
| 353 | + break; | |
| 354 | + | |
| 355 | + case 0x20: /* INVOFF */ | |
| 356 | + s->invert = 0; | |
| 357 | + break; | |
| 358 | + case 0x21: /* INVON */ | |
| 359 | + s->invert = 1; | |
| 360 | + break; | |
| 361 | + | |
| 362 | + case 0x22: /* APOFF */ | |
| 363 | + case 0x23: /* APON */ | |
| 364 | + goto bad_cmd; | |
| 365 | + | |
| 366 | + case 0x25: /* WRCNTR */ | |
| 367 | + if (s->pm < 0) | |
| 368 | + s->pm = 1; | |
| 369 | + goto bad_cmd; | |
| 370 | + | |
| 371 | + case 0x26: /* GAMSET */ | |
| 372 | + if (!s->pm) | |
| 373 | + s->gamma = ffs(s->param[0] & 0xf) - 1; | |
| 374 | + else if (s->pm < 0) | |
| 375 | + s->pm = 1; | |
| 376 | + break; | |
| 377 | + | |
| 378 | + case 0x28: /* DISPOFF */ | |
| 379 | + s->onoff = 0; | |
| 380 | + fprintf(stderr, "%s: Display off\n", __FUNCTION__); | |
| 381 | + break; | |
| 382 | + case 0x29: /* DISPON */ | |
| 383 | + s->onoff = 1; | |
| 384 | + fprintf(stderr, "%s: Display on\n", __FUNCTION__); | |
| 385 | + break; | |
| 386 | + | |
| 387 | + case 0x2a: /* CASET */ | |
| 388 | + case 0x2b: /* RASET */ | |
| 389 | + case 0x2c: /* RAMWR */ | |
| 390 | + case 0x2d: /* RGBSET */ | |
| 391 | + case 0x2e: /* RAMRD */ | |
| 392 | + case 0x30: /* PTLAR */ | |
| 393 | + case 0x33: /* SCRLAR */ | |
| 394 | + goto bad_cmd; | |
| 395 | + | |
| 396 | + case 0x34: /* TEOFF */ | |
| 397 | + s->te = 0; | |
| 398 | + break; | |
| 399 | + case 0x35: /* TEON */ | |
| 400 | + if (!s->pm) | |
| 401 | + s->te = 1; | |
| 402 | + else if (s->pm < 0) | |
| 403 | + s->pm = 1; | |
| 404 | + break; | |
| 405 | + | |
| 406 | + case 0x36: /* MADCTR */ | |
| 407 | + goto bad_cmd; | |
| 408 | + | |
| 409 | + case 0x37: /* VSCSAD */ | |
| 410 | + s->partial = 0; | |
| 411 | + s->normal = 0; | |
| 412 | + s->vscr = 1; | |
| 413 | + break; | |
| 414 | + | |
| 415 | + case 0x38: /* IDMOFF */ | |
| 416 | + case 0x39: /* IDMON */ | |
| 417 | + case 0x3a: /* COLMOD */ | |
| 418 | + goto bad_cmd; | |
| 419 | + | |
| 420 | + case 0xb0: /* CLKINT / DISCTL */ | |
| 421 | + case 0xb1: /* CLKEXT */ | |
| 422 | + if (s->pm < 0) | |
| 423 | + s->pm = 2; | |
| 424 | + break; | |
| 425 | + | |
| 426 | + case 0xb4: /* FRMSEL */ | |
| 427 | + break; | |
| 428 | + | |
| 429 | + case 0xb5: /* FRM8SEL */ | |
| 430 | + case 0xb6: /* TMPRNG / INIESC */ | |
| 431 | + case 0xb7: /* TMPHIS / NOP2 */ | |
| 432 | + case 0xb8: /* TMPREAD / MADCTL */ | |
| 433 | + case 0xba: /* DISTCTR */ | |
| 434 | + case 0xbb: /* EPVOL */ | |
| 435 | + goto bad_cmd; | |
| 436 | + | |
| 437 | + case 0xbd: /* Unknown */ | |
| 438 | + s->p = 0; | |
| 439 | + s->resp[0] = 0; | |
| 440 | + s->resp[1] = 1; | |
| 441 | + break; | |
| 442 | + | |
| 443 | + case 0xc2: /* IFMOD */ | |
| 444 | + if (s->pm < 0) | |
| 445 | + s->pm = 2; | |
| 446 | + break; | |
| 447 | + | |
| 448 | + case 0xc6: /* PWRCTL */ | |
| 449 | + case 0xc7: /* PPWRCTL */ | |
| 450 | + case 0xd0: /* EPWROUT */ | |
| 451 | + case 0xd1: /* EPWRIN */ | |
| 452 | + case 0xd4: /* RDEV */ | |
| 453 | + case 0xd5: /* RDRR */ | |
| 454 | + goto bad_cmd; | |
| 455 | + | |
| 456 | + case 0xda: /* RDID1 */ | |
| 457 | + s->p = 0; | |
| 458 | + s->resp[0] = (s->id >> 16) & 0xff; | |
| 459 | + break; | |
| 460 | + case 0xdb: /* RDID2 */ | |
| 461 | + s->p = 0; | |
| 462 | + s->resp[0] = (s->id >> 8) & 0xff; | |
| 463 | + break; | |
| 464 | + case 0xdc: /* RDID3 */ | |
| 465 | + s->p = 0; | |
| 466 | + s->resp[0] = (s->id >> 0) & 0xff; | |
| 467 | + break; | |
| 468 | + | |
| 469 | + default: | |
| 470 | + bad_cmd: | |
| 471 | + fprintf(stderr, "%s: unknown command %02x\n", __FUNCTION__, s->cmd); | |
| 472 | + break; | |
| 473 | + } | |
| 474 | + | |
| 475 | + return ret; | |
| 476 | +} | |
| 477 | + | |
| 478 | +static void *mipid_init(void) | |
| 479 | +{ | |
| 480 | + struct mipid_s *s = (struct mipid_s *) qemu_mallocz(sizeof(*s)); | |
| 481 | + | |
| 482 | + s->id = 0x838f03; | |
| 483 | + mipid_reset(s); | |
| 484 | + | |
| 485 | + return s; | |
| 486 | +} | |
| 487 | + | |
| 488 | +static void n800_spi_setup(struct n800_s *s) | |
| 489 | +{ | |
| 490 | + void *tsc2301 = s->ts->opaque; | |
| 491 | + void *mipid = mipid_init(); | |
| 492 | + | |
| 493 | + omap_mcspi_attach(s->cpu->mcspi[0], tsc210x_txrx, tsc2301, 0); | |
| 494 | + omap_mcspi_attach(s->cpu->mcspi[0], mipid_txrx, mipid, 1); | |
| 495 | +} | |
| 496 | + | |
| 497 | +/* This task is normally performed by the bootloader. If we're loading | |
| 498 | + * a kernel directly, we need to enable the Blizzard ourselves. */ | |
| 499 | +static void n800_dss_init(struct rfbi_chip_s *chip) | |
| 500 | +{ | |
| 501 | + uint8_t *fb_blank; | |
| 502 | + | |
| 503 | + chip->write(chip->opaque, 0, 0x2a); /* LCD Width register */ | |
| 504 | + chip->write(chip->opaque, 1, 0x64); | |
| 505 | + chip->write(chip->opaque, 0, 0x2c); /* LCD HNDP register */ | |
| 506 | + chip->write(chip->opaque, 1, 0x1e); | |
| 507 | + chip->write(chip->opaque, 0, 0x2e); /* LCD Height 0 register */ | |
| 508 | + chip->write(chip->opaque, 1, 0xe0); | |
| 509 | + chip->write(chip->opaque, 0, 0x30); /* LCD Height 1 register */ | |
| 510 | + chip->write(chip->opaque, 1, 0x01); | |
| 511 | + chip->write(chip->opaque, 0, 0x32); /* LCD VNDP register */ | |
| 512 | + chip->write(chip->opaque, 1, 0x06); | |
| 513 | + chip->write(chip->opaque, 0, 0x68); /* Display Mode register */ | |
| 514 | + chip->write(chip->opaque, 1, 1); /* Enable bit */ | |
| 515 | + | |
| 516 | + chip->write(chip->opaque, 0, 0x6c); | |
| 517 | + chip->write(chip->opaque, 1, 0x00); /* Input X Start Position */ | |
| 518 | + chip->write(chip->opaque, 1, 0x00); /* Input X Start Position */ | |
| 519 | + chip->write(chip->opaque, 1, 0x00); /* Input Y Start Position */ | |
| 520 | + chip->write(chip->opaque, 1, 0x00); /* Input Y Start Position */ | |
| 521 | + chip->write(chip->opaque, 1, 0x1f); /* Input X End Position */ | |
| 522 | + chip->write(chip->opaque, 1, 0x03); /* Input X End Position */ | |
| 523 | + chip->write(chip->opaque, 1, 0xdf); /* Input Y End Position */ | |
| 524 | + chip->write(chip->opaque, 1, 0x01); /* Input Y End Position */ | |
| 525 | + chip->write(chip->opaque, 1, 0x00); /* Output X Start Position */ | |
| 526 | + chip->write(chip->opaque, 1, 0x00); /* Output X Start Position */ | |
| 527 | + chip->write(chip->opaque, 1, 0x00); /* Output Y Start Position */ | |
| 528 | + chip->write(chip->opaque, 1, 0x00); /* Output Y Start Position */ | |
| 529 | + chip->write(chip->opaque, 1, 0x1f); /* Output X End Position */ | |
| 530 | + chip->write(chip->opaque, 1, 0x03); /* Output X End Position */ | |
| 531 | + chip->write(chip->opaque, 1, 0xdf); /* Output Y End Position */ | |
| 532 | + chip->write(chip->opaque, 1, 0x01); /* Output Y End Position */ | |
| 533 | + chip->write(chip->opaque, 1, 0x01); /* Input Data Format */ | |
| 534 | + chip->write(chip->opaque, 1, 0x01); /* Data Source Select */ | |
| 535 | + | |
| 536 | + fb_blank = memset(qemu_malloc(800 * 480 * 2), 0xff, 800 * 480 * 2); | |
| 537 | + /* Display Memory Data Port */ | |
| 538 | + chip->block(chip->opaque, 1, fb_blank, 800 * 480 * 2, 800); | |
| 539 | + free(fb_blank); | |
| 540 | +} | |
| 541 | + | |
| 542 | +static void n800_dss_setup(struct n800_s *s, DisplayState *ds) | |
| 543 | +{ | |
| 544 | + s->blizzard.opaque = s1d13745_init(0, ds); | |
| 545 | + s->blizzard.block = s1d13745_write_block; | |
| 546 | + s->blizzard.write = s1d13745_write; | |
| 547 | + s->blizzard.read = s1d13745_read; | |
| 548 | + | |
| 549 | + omap_rfbi_attach(s->cpu->dss, 0, &s->blizzard); | |
| 550 | +} | |
| 551 | + | |
| 552 | +static void n800_cbus_setup(struct n800_s *s) | |
| 553 | +{ | |
| 554 | + qemu_irq dat_out = omap2_gpio_in_get(s->cpu->gpif, N8X0_CBUS_DAT_GPIO)[0]; | |
| 555 | + qemu_irq retu_irq = omap2_gpio_in_get(s->cpu->gpif, N8X0_RETU_GPIO)[0]; | |
| 556 | + qemu_irq tahvo_irq = omap2_gpio_in_get(s->cpu->gpif, N8X0_TAHVO_GPIO)[0]; | |
| 557 | + | |
| 558 | + struct cbus_s *cbus = cbus_init(dat_out); | |
| 559 | + | |
| 560 | + omap2_gpio_out_set(s->cpu->gpif, N8X0_CBUS_CLK_GPIO, cbus->clk); | |
| 561 | + omap2_gpio_out_set(s->cpu->gpif, N8X0_CBUS_DAT_GPIO, cbus->dat); | |
| 562 | + omap2_gpio_out_set(s->cpu->gpif, N8X0_CBUS_SEL_GPIO, cbus->sel); | |
| 563 | + | |
| 564 | + cbus_attach(cbus, s->retu = retu_init(retu_irq, 1)); | |
| 565 | + cbus_attach(cbus, s->tahvo = tahvo_init(tahvo_irq, 1)); | |
| 566 | +} | |
| 567 | + | |
| 568 | +/* This task is normally performed by the bootloader. If we're loading | |
| 569 | + * a kernel directly, we need to set up GPMC mappings ourselves. */ | |
| 570 | +static void n800_gpmc_init(struct n800_s *s) | |
| 571 | +{ | |
| 572 | + uint32_t config7 = | |
| 573 | + (0xf << 8) | /* MASKADDRESS */ | |
| 574 | + (1 << 6) | /* CSVALID */ | |
| 575 | + (4 << 0); /* BASEADDRESS */ | |
| 576 | + | |
| 577 | + cpu_physical_memory_write(0x6800a078, /* GPMC_CONFIG7_0 */ | |
| 578 | + (void *) &config7, sizeof(config7)); | |
| 579 | +} | |
| 580 | + | |
| 581 | +/* Setup sequence done by the bootloader */ | |
| 582 | +static void n800_boot_init(void *opaque) | |
| 583 | +{ | |
| 584 | + struct n800_s *s = (struct n800_s *) opaque; | |
| 585 | + uint32_t buf; | |
| 586 | + | |
| 587 | + /* PRCM setup */ | |
| 588 | +#define omap_writel(addr, val) \ | |
| 589 | + buf = (val); \ | |
| 590 | + cpu_physical_memory_write(addr, (void *) &buf, sizeof(buf)) | |
| 591 | + | |
| 592 | + omap_writel(0x48008060, 0x41); /* PRCM_CLKSRC_CTRL */ | |
| 593 | + omap_writel(0x48008070, 1); /* PRCM_CLKOUT_CTRL */ | |
| 594 | + omap_writel(0x48008078, 0); /* PRCM_CLKEMUL_CTRL */ | |
| 595 | + omap_writel(0x48008090, 0); /* PRCM_VOLTSETUP */ | |
| 596 | + omap_writel(0x48008094, 0); /* PRCM_CLKSSETUP */ | |
| 597 | + omap_writel(0x48008098, 0); /* PRCM_POLCTRL */ | |
| 598 | + omap_writel(0x48008140, 2); /* CM_CLKSEL_MPU */ | |
| 599 | + omap_writel(0x48008148, 0); /* CM_CLKSTCTRL_MPU */ | |
| 600 | + omap_writel(0x48008158, 1); /* RM_RSTST_MPU */ | |
| 601 | + omap_writel(0x480081c8, 0x15); /* PM_WKDEP_MPU */ | |
| 602 | + omap_writel(0x480081d4, 0x1d4); /* PM_EVGENCTRL_MPU */ | |
| 603 | + omap_writel(0x480081d8, 0); /* PM_EVEGENONTIM_MPU */ | |
| 604 | + omap_writel(0x480081dc, 0); /* PM_EVEGENOFFTIM_MPU */ | |
| 605 | + omap_writel(0x480081e0, 0xc); /* PM_PWSTCTRL_MPU */ | |
| 606 | + omap_writel(0x48008200, 0x047e7ff7); /* CM_FCLKEN1_CORE */ | |
| 607 | + omap_writel(0x48008204, 0x00000004); /* CM_FCLKEN2_CORE */ | |
| 608 | + omap_writel(0x48008210, 0x047e7ff1); /* CM_ICLKEN1_CORE */ | |
| 609 | + omap_writel(0x48008214, 0x00000004); /* CM_ICLKEN2_CORE */ | |
| 610 | + omap_writel(0x4800821c, 0x00000000); /* CM_ICLKEN4_CORE */ | |
| 611 | + omap_writel(0x48008230, 0); /* CM_AUTOIDLE1_CORE */ | |
| 612 | + omap_writel(0x48008234, 0); /* CM_AUTOIDLE2_CORE */ | |
| 613 | + omap_writel(0x48008238, 7); /* CM_AUTOIDLE3_CORE */ | |
| 614 | + omap_writel(0x4800823c, 0); /* CM_AUTOIDLE4_CORE */ | |
| 615 | + omap_writel(0x48008240, 0x04360626); /* CM_CLKSEL1_CORE */ | |
| 616 | + omap_writel(0x48008244, 0x00000014); /* CM_CLKSEL2_CORE */ | |
| 617 | + omap_writel(0x48008248, 0); /* CM_CLKSTCTRL_CORE */ | |
| 618 | + omap_writel(0x48008300, 0x00000000); /* CM_FCLKEN_GFX */ | |
| 619 | + omap_writel(0x48008310, 0x00000000); /* CM_ICLKEN_GFX */ | |
| 620 | + omap_writel(0x48008340, 0x00000001); /* CM_CLKSEL_GFX */ | |
| 621 | + omap_writel(0x48008400, 0x00000004); /* CM_FCLKEN_WKUP */ | |
| 622 | + omap_writel(0x48008410, 0x00000004); /* CM_ICLKEN_WKUP */ | |
| 623 | + omap_writel(0x48008440, 0x00000000); /* CM_CLKSEL_WKUP */ | |
| 624 | + omap_writel(0x48008500, 0x000000cf); /* CM_CLKEN_PLL */ | |
| 625 | + omap_writel(0x48008530, 0x0000000c); /* CM_AUTOIDLE_PLL */ | |
| 626 | + omap_writel(0x48008540, /* CM_CLKSEL1_PLL */ | |
| 627 | + (0x78 << 12) | (6 << 8)); | |
| 628 | + omap_writel(0x48008544, 2); /* CM_CLKSEL2_PLL */ | |
| 629 | + | |
| 630 | + /* GPMC setup */ | |
| 631 | + n800_gpmc_init(s); | |
| 632 | + | |
| 633 | + /* Video setup */ | |
| 634 | + n800_dss_init(&s->blizzard); | |
| 635 | + | |
| 636 | + /* CPU setup */ | |
| 637 | + s->cpu->env->regs[15] = s->cpu->env->boot_info->loader_start; | |
| 638 | + s->cpu->env->GE = 0x5; | |
| 639 | +} | |
| 640 | + | |
| 641 | +#define OMAP_TAG_NOKIA_BT 0x4e01 | |
| 642 | +#define OMAP_TAG_WLAN_CX3110X 0x4e02 | |
| 643 | +#define OMAP_TAG_CBUS 0x4e03 | |
| 644 | +#define OMAP_TAG_EM_ASIC_BB5 0x4e04 | |
| 645 | + | |
| 646 | +static int n800_atag_setup(struct arm_boot_info *info, void *p) | |
| 647 | +{ | |
| 648 | + uint8_t *b; | |
| 649 | + uint16_t *w; | |
| 650 | + uint32_t *l; | |
| 651 | + | |
| 652 | + w = p; | |
| 653 | + | |
| 654 | + stw_raw(w ++, OMAP_TAG_UART); /* u16 tag */ | |
| 655 | + stw_raw(w ++, 4); /* u16 len */ | |
| 656 | + stw_raw(w ++, (1 << 2) | (1 << 1) | (1 << 0)); /* uint enabled_uarts */ | |
| 657 | + w ++; | |
| 658 | + | |
| 659 | + stw_raw(w ++, OMAP_TAG_EM_ASIC_BB5); /* u16 tag */ | |
| 660 | + stw_raw(w ++, 4); /* u16 len */ | |
| 661 | + stw_raw(w ++, N8X0_RETU_GPIO); /* s16 retu_irq_gpio */ | |
| 662 | + stw_raw(w ++, N8X0_TAHVO_GPIO); /* s16 tahvo_irq_gpio */ | |
| 663 | + | |
| 664 | + stw_raw(w ++, OMAP_TAG_CBUS); /* u16 tag */ | |
| 665 | + stw_raw(w ++, 8); /* u16 len */ | |
| 666 | + stw_raw(w ++, N8X0_CBUS_CLK_GPIO); /* s16 clk_gpio */ | |
| 667 | + stw_raw(w ++, N8X0_CBUS_DAT_GPIO); /* s16 dat_gpio */ | |
| 668 | + stw_raw(w ++, N8X0_CBUS_SEL_GPIO); /* s16 sel_gpio */ | |
| 669 | + w ++; | |
| 670 | + | |
| 671 | + stw_raw(w ++, OMAP_TAG_GPIO_SWITCH); /* u16 tag */ | |
| 672 | + stw_raw(w ++, 20); /* u16 len */ | |
| 673 | + strcpy((void *) w, "bat_cover"); /* char name[12] */ | |
| 674 | + w += 6; | |
| 675 | + stw_raw(w ++, N800_BAT_COVER_GPIO); /* u16 gpio */ | |
| 676 | + stw_raw(w ++, 0x01); | |
| 677 | + stw_raw(w ++, 0); | |
| 678 | + stw_raw(w ++, 0); | |
| 679 | + | |
| 680 | + stw_raw(w ++, OMAP_TAG_GPIO_SWITCH); /* u16 tag */ | |
| 681 | + stw_raw(w ++, 20); /* u16 len */ | |
| 682 | + strcpy((void *) w, "cam_act"); /* char name[12] */ | |
| 683 | + w += 6; | |
| 684 | + stw_raw(w ++, N800_CAM_ACT_GPIO); /* u16 gpio */ | |
| 685 | + stw_raw(w ++, 0x20); | |
| 686 | + stw_raw(w ++, 0); | |
| 687 | + stw_raw(w ++, 0); | |
| 688 | + | |
| 689 | + stw_raw(w ++, OMAP_TAG_GPIO_SWITCH); /* u16 tag */ | |
| 690 | + stw_raw(w ++, 20); /* u16 len */ | |
| 691 | + strcpy((void *) w, "cam_turn"); /* char name[12] */ | |
| 692 | + w += 6; | |
| 693 | + stw_raw(w ++, N800_CAM_TURN_GPIO); /* u16 gpio */ | |
| 694 | + stw_raw(w ++, 0x21); | |
| 695 | + stw_raw(w ++, 0); | |
| 696 | + stw_raw(w ++, 0); | |
| 697 | + | |
| 698 | + stw_raw(w ++, OMAP_TAG_GPIO_SWITCH); /* u16 tag */ | |
| 699 | + stw_raw(w ++, 20); /* u16 len */ | |
| 700 | + strcpy((void *) w, "headphone"); /* char name[12] */ | |
| 701 | + w += 6; | |
| 702 | + stw_raw(w ++, N800_HEADPHONE_GPIO); /* u16 gpio */ | |
| 703 | + stw_raw(w ++, 0x11); | |
| 704 | + stw_raw(w ++, 0); | |
| 705 | + stw_raw(w ++, 0); | |
| 706 | + | |
| 707 | + stw_raw(w ++, OMAP_TAG_NOKIA_BT); /* u16 tag */ | |
| 708 | + stw_raw(w ++, 12); /* u16 len */ | |
| 709 | + b = (void *) w; | |
| 710 | + stb_raw(b ++, 0x01); /* u8 chip_type (CSR) */ | |
| 711 | + stb_raw(b ++, N800_BT_WKUP_GPIO); /* u8 bt_wakeup_gpio */ | |
| 712 | + stb_raw(b ++, N8X0_BT_HOST_WKUP_GPIO); /* u8 host_wakeup_gpio */ | |
| 713 | + stb_raw(b ++, N800_BT_RESET_GPIO); /* u8 reset_gpio */ | |
| 714 | + stb_raw(b ++, 1); /* u8 bt_uart */ | |
| 715 | + memset(b, 0, 6); /* u8 bd_addr[6] */ | |
| 716 | + b += 6; | |
| 717 | + stb_raw(b ++, 0x02); /* u8 bt_sysclk (38.4) */ | |
| 718 | + w = (void *) b; | |
| 719 | + | |
| 720 | + stw_raw(w ++, OMAP_TAG_WLAN_CX3110X); /* u16 tag */ | |
| 721 | + stw_raw(w ++, 8); /* u16 len */ | |
| 722 | + stw_raw(w ++, 0x25); /* u8 chip_type */ | |
| 723 | + stw_raw(w ++, N800_WLAN_PWR_GPIO); /* s16 power_gpio */ | |
| 724 | + stw_raw(w ++, N800_WLAN_IRQ_GPIO); /* s16 irq_gpio */ | |
| 725 | + stw_raw(w ++, -1); /* s16 spi_cs_gpio */ | |
| 726 | + | |
| 727 | + stw_raw(w ++, OMAP_TAG_MMC); /* u16 tag */ | |
| 728 | + stw_raw(w ++, 16); /* u16 len */ | |
| 729 | + stw_raw(w ++, 0xf); /* unsigned flags */ | |
| 730 | + stw_raw(w ++, -1); /* s16 power_pin */ | |
| 731 | + stw_raw(w ++, -1); /* s16 switch_pin */ | |
| 732 | + stw_raw(w ++, -1); /* s16 wp_pin */ | |
| 733 | + stw_raw(w ++, 0); /* unsigned flags */ | |
| 734 | + stw_raw(w ++, 0); /* s16 power_pin */ | |
| 735 | + stw_raw(w ++, 0); /* s16 switch_pin */ | |
| 736 | + stw_raw(w ++, 0); /* s16 wp_pin */ | |
| 737 | + | |
| 738 | + stw_raw(w ++, OMAP_TAG_TEA5761); /* u16 tag */ | |
| 739 | + stw_raw(w ++, 4); /* u16 len */ | |
| 740 | + stw_raw(w ++, N800_TEA5761_CS_GPIO); /* u16 enable_gpio */ | |
| 741 | + w ++; | |
| 742 | + | |
| 743 | + stw_raw(w ++, OMAP_TAG_PARTITION); /* u16 tag */ | |
| 744 | + stw_raw(w ++, 28); /* u16 len */ | |
| 745 | + strcpy((void *) w, "bootloader"); /* char name[16] */ | |
| 746 | + l = (void *) (w + 8); | |
| 747 | + stl_raw(l ++, 0x00020000); /* unsigned int size */ | |
| 748 | + stl_raw(l ++, 0x00000000); /* unsigned int offset */ | |
| 749 | + stl_raw(l ++, 0x3); /* unsigned int mask_flags */ | |
| 750 | + w = (void *) l; | |
| 751 | + | |
| 752 | + stw_raw(w ++, OMAP_TAG_PARTITION); /* u16 tag */ | |
| 753 | + stw_raw(w ++, 28); /* u16 len */ | |
| 754 | + strcpy((void *) w, "config"); /* char name[16] */ | |
| 755 | + l = (void *) (w + 8); | |
| 756 | + stl_raw(l ++, 0x00060000); /* unsigned int size */ | |
| 757 | + stl_raw(l ++, 0x00020000); /* unsigned int offset */ | |
| 758 | + stl_raw(l ++, 0x0); /* unsigned int mask_flags */ | |
| 759 | + w = (void *) l; | |
| 760 | + | |
| 761 | + stw_raw(w ++, OMAP_TAG_PARTITION); /* u16 tag */ | |
| 762 | + stw_raw(w ++, 28); /* u16 len */ | |
| 763 | + strcpy((void *) w, "kernel"); /* char name[16] */ | |
| 764 | + l = (void *) (w + 8); | |
| 765 | + stl_raw(l ++, 0x00200000); /* unsigned int size */ | |
| 766 | + stl_raw(l ++, 0x00080000); /* unsigned int offset */ | |
| 767 | + stl_raw(l ++, 0x0); /* unsigned int mask_flags */ | |
| 768 | + w = (void *) l; | |
| 769 | + | |
| 770 | + stw_raw(w ++, OMAP_TAG_PARTITION); /* u16 tag */ | |
| 771 | + stw_raw(w ++, 28); /* u16 len */ | |
| 772 | + strcpy((void *) w, "initfs"); /* char name[16] */ | |
| 773 | + l = (void *) (w + 8); | |
| 774 | + stl_raw(l ++, 0x00200000); /* unsigned int size */ | |
| 775 | + stl_raw(l ++, 0x00280000); /* unsigned int offset */ | |
| 776 | + stl_raw(l ++, 0x3); /* unsigned int mask_flags */ | |
| 777 | + w = (void *) l; | |
| 778 | + | |
| 779 | + stw_raw(w ++, OMAP_TAG_PARTITION); /* u16 tag */ | |
| 780 | + stw_raw(w ++, 28); /* u16 len */ | |
| 781 | + strcpy((void *) w, "rootfs"); /* char name[16] */ | |
| 782 | + l = (void *) (w + 8); | |
| 783 | + stl_raw(l ++, 0x0fb80000); /* unsigned int size */ | |
| 784 | + stl_raw(l ++, 0x00480000); /* unsigned int offset */ | |
| 785 | + stl_raw(l ++, 0x3); /* unsigned int mask_flags */ | |
| 786 | + w = (void *) l; | |
| 787 | + | |
| 788 | + stw_raw(w ++, OMAP_TAG_BOOT_REASON); /* u16 tag */ | |
| 789 | + stw_raw(w ++, 12); /* u16 len */ | |
| 790 | +#if 0 | |
| 791 | + strcpy((void *) w, "por"); /* char reason_str[12] */ | |
| 792 | + strcpy((void *) w, "charger"); /* char reason_str[12] */ | |
| 793 | + strcpy((void *) w, "32wd_to"); /* char reason_str[12] */ | |
| 794 | + strcpy((void *) w, "sw_rst"); /* char reason_str[12] */ | |
| 795 | + strcpy((void *) w, "mbus"); /* char reason_str[12] */ | |
| 796 | + strcpy((void *) w, "unknown"); /* char reason_str[12] */ | |
| 797 | + strcpy((void *) w, "swdg_to"); /* char reason_str[12] */ | |
| 798 | + strcpy((void *) w, "sec_vio"); /* char reason_str[12] */ | |
| 799 | + strcpy((void *) w, "pwr_key"); /* char reason_str[12] */ | |
| 800 | + strcpy((void *) w, "rtc_alarm"); /* char reason_str[12] */ | |
| 801 | +#else | |
| 802 | + strcpy((void *) w, "pwr_key"); /* char reason_str[12] */ | |
| 803 | +#endif | |
| 804 | + w += 6; | |
| 805 | + | |
| 806 | +#if 0 /* N810 */ | |
| 807 | + stw_raw(w ++, OMAP_TAG_VERSION_STR); /* u16 tag */ | |
| 808 | + stw_raw(w ++, 24); /* u16 len */ | |
| 809 | + strcpy((void *) w, "product"); /* char component[12] */ | |
| 810 | + w += 6; | |
| 811 | + strcpy((void *) w, "RX-44"); /* char version[12] */ | |
| 812 | + w += 6; | |
| 813 | + | |
| 814 | + stw_raw(w ++, OMAP_TAG_VERSION_STR); /* u16 tag */ | |
| 815 | + stw_raw(w ++, 24); /* u16 len */ | |
| 816 | + strcpy((void *) w, "hw-build"); /* char component[12] */ | |
| 817 | + w += 6; | |
| 818 | + strcpy((void *) w, "QEMU"); /* char version[12] */ | |
| 819 | + w += 6; | |
| 820 | + | |
| 821 | + stw_raw(w ++, OMAP_TAG_VERSION_STR); /* u16 tag */ | |
| 822 | + stw_raw(w ++, 24); /* u16 len */ | |
| 823 | + strcpy((void *) w, "nolo"); /* char component[12] */ | |
| 824 | + w += 6; | |
| 825 | + strcpy((void *) w, "1.1.10-qemu"); /* char version[12] */ | |
| 826 | + w += 6; | |
| 827 | +#else | |
| 828 | + stw_raw(w ++, OMAP_TAG_VERSION_STR); /* u16 tag */ | |
| 829 | + stw_raw(w ++, 24); /* u16 len */ | |
| 830 | + strcpy((void *) w, "product"); /* char component[12] */ | |
| 831 | + w += 6; | |
| 832 | + strcpy((void *) w, "RX-34"); /* char version[12] */ | |
| 833 | + w += 6; | |
| 834 | + | |
| 835 | + stw_raw(w ++, OMAP_TAG_VERSION_STR); /* u16 tag */ | |
| 836 | + stw_raw(w ++, 24); /* u16 len */ | |
| 837 | + strcpy((void *) w, "hw-build"); /* char component[12] */ | |
| 838 | + w += 6; | |
| 839 | + strcpy((void *) w, "QEMU"); /* char version[12] */ | |
| 840 | + w += 6; | |
| 841 | + | |
| 842 | + stw_raw(w ++, OMAP_TAG_VERSION_STR); /* u16 tag */ | |
| 843 | + stw_raw(w ++, 24); /* u16 len */ | |
| 844 | + strcpy((void *) w, "nolo"); /* char component[12] */ | |
| 845 | + w += 6; | |
| 846 | + strcpy((void *) w, "1.1.6-qemu"); /* char version[12] */ | |
| 847 | + w += 6; | |
| 848 | +#endif | |
| 849 | + | |
| 850 | + stw_raw(w ++, OMAP_TAG_LCD); /* u16 tag */ | |
| 851 | + stw_raw(w ++, 36); /* u16 len */ | |
| 852 | + strcpy((void *) w, "QEMU LCD panel"); /* char panel_name[16] */ | |
| 853 | + w += 8; | |
| 854 | + strcpy((void *) w, "blizzard"); /* char ctrl_name[16] */ | |
| 855 | + w += 8; | |
| 856 | + stw_raw(w ++, 5); /* TODO s16 nreset_gpio */ | |
| 857 | + stw_raw(w ++, 16); /* u8 data_lines */ | |
| 858 | + | |
| 859 | + return (void *) w - p; | |
| 860 | +} | |
| 861 | + | |
| 862 | +static struct arm_boot_info n800_binfo = { | |
| 863 | + .loader_start = OMAP2_Q2_BASE, | |
| 864 | + /* Actually two chips of 0x4000000 bytes each */ | |
| 865 | + .ram_size = 0x08000000, | |
| 866 | + .board_id = 0x4f7, | |
| 867 | + .atag_board = n800_atag_setup, | |
| 868 | +}; | |
| 869 | + | |
| 870 | +static void n800_init(int ram_size, int vga_ram_size, | |
| 871 | + const char *boot_device, DisplayState *ds, | |
| 872 | + const char *kernel_filename, const char *kernel_cmdline, | |
| 873 | + const char *initrd_filename, const char *cpu_model) | |
| 874 | +{ | |
| 875 | + struct n800_s *s = (struct n800_s *) qemu_mallocz(sizeof(*s)); | |
| 876 | + int sdram_size = n800_binfo.ram_size; | |
| 877 | + int onenandram_size = 0x00010000; | |
| 878 | + | |
| 879 | + if (ram_size < sdram_size + onenandram_size + OMAP242X_SRAM_SIZE) { | |
| 880 | + fprintf(stderr, "This architecture uses %i bytes of memory\n", | |
| 881 | + sdram_size + onenandram_size + OMAP242X_SRAM_SIZE); | |
| 882 | + exit(1); | |
| 883 | + } | |
| 884 | + | |
| 885 | + s->cpu = omap2420_mpu_init(sdram_size, NULL, cpu_model); | |
| 886 | + | |
| 887 | + n800_gpio_setup(s); | |
| 888 | + n8x0_nand_setup(s); | |
| 889 | + n800_i2c_setup(s); | |
| 890 | + n800_tsc_setup(s); | |
| 891 | + n800_spi_setup(s); | |
| 892 | + n800_dss_setup(s, ds); | |
| 893 | + n800_cbus_setup(s); | |
| 894 | + | |
| 895 | + /* Setup initial (reset) machine state */ | |
| 896 | + | |
| 897 | + /* Start at the OneNAND bootloader. */ | |
| 898 | + s->cpu->env->regs[15] = 0; | |
| 899 | + | |
| 900 | + if (kernel_filename) { | |
| 901 | + /* Or at the linux loader. */ | |
| 902 | + n800_binfo.kernel_filename = kernel_filename; | |
| 903 | + n800_binfo.kernel_cmdline = kernel_cmdline; | |
| 904 | + n800_binfo.initrd_filename = initrd_filename; | |
| 905 | + arm_load_kernel(s->cpu->env, &n800_binfo); | |
| 906 | + | |
| 907 | + qemu_register_reset(n800_boot_init, s); | |
| 908 | + n800_boot_init(s); | |
| 909 | + } | |
| 910 | + | |
| 911 | + dpy_resize(ds, 800, 480); | |
| 912 | +} | |
| 913 | + | |
| 914 | +QEMUMachine n800_machine = { | |
| 915 | + "n800", | |
| 916 | + "Nokia N800 aka. RX-34 tablet (OMAP2420)", | |
| 917 | + n800_init, | |
| 918 | +}; | ... | ... |
hw/omap2.c
| ... | ... | @@ -3496,7 +3496,7 @@ struct omap_mpu_state_s *omap2420_mpu_init(unsigned long sdram_size, |
| 3496 | 3496 | { |
| 3497 | 3497 | struct omap_mpu_state_s *s = (struct omap_mpu_state_s *) |
| 3498 | 3498 | qemu_mallocz(sizeof(struct omap_mpu_state_s)); |
| 3499 | - ram_addr_t sram_base, q3_base; | |
| 3499 | + ram_addr_t sram_base, q2_base; | |
| 3500 | 3500 | qemu_irq *cpu_irq; |
| 3501 | 3501 | qemu_irq dma_irqs[4]; |
| 3502 | 3502 | omap_clk gpio_clks[4]; |
| ... | ... | @@ -3520,7 +3520,7 @@ struct omap_mpu_state_s *omap2420_mpu_init(unsigned long sdram_size, |
| 3520 | 3520 | |
| 3521 | 3521 | /* Memory-mapped stuff */ |
| 3522 | 3522 | cpu_register_physical_memory(OMAP2_Q2_BASE, s->sdram_size, |
| 3523 | - (q3_base = qemu_ram_alloc(s->sdram_size)) | IO_MEM_RAM); | |
| 3523 | + (q2_base = qemu_ram_alloc(s->sdram_size)) | IO_MEM_RAM); | |
| 3524 | 3524 | cpu_register_physical_memory(OMAP2_SRAM_BASE, s->sram_size, |
| 3525 | 3525 | (sram_base = qemu_ram_alloc(s->sram_size)) | IO_MEM_RAM); |
| 3526 | 3526 | ... | ... |
hw/onenand.c
0 → 100644
| 1 | +/* | |
| 2 | + * OneNAND flash memories emulation. | |
| 3 | + * | |
| 4 | + * Copyright (C) 2008 Nokia Corporation | |
| 5 | + * Written by Andrzej Zaborowski <andrew@openedhand.com> | |
| 6 | + * | |
| 7 | + * This program is free software; you can redistribute it and/or | |
| 8 | + * modify it under the terms of the GNU General Public License as | |
| 9 | + * published by the Free Software Foundation; either version 2 or | |
| 10 | + * (at your option) version 3 of the License. | |
| 11 | + * | |
| 12 | + * This program is distributed in the hope that it will be useful, | |
| 13 | + * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
| 14 | + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
| 15 | + * GNU General Public License for more details. | |
| 16 | + * | |
| 17 | + * You should have received a copy of the GNU General Public License | |
| 18 | + * along with this program; if not, write to the Free Software | |
| 19 | + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, | |
| 20 | + * MA 02111-1307 USA | |
| 21 | + */ | |
| 22 | + | |
| 23 | +#include "qemu-common.h" | |
| 24 | +#include "flash.h" | |
| 25 | +#include "irq.h" | |
| 26 | +#include "sysemu.h" | |
| 27 | +#include "block.h" | |
| 28 | + | |
| 29 | +/* 11 for 2kB-page OneNAND ("2nd generation") and 10 for 1kB-page chips */ | |
| 30 | +#define PAGE_SHIFT 11 | |
| 31 | + | |
| 32 | +/* Fixed */ | |
| 33 | +#define BLOCK_SHIFT (PAGE_SHIFT + 6) | |
| 34 | + | |
| 35 | +struct onenand_s { | |
| 36 | + uint32_t id; | |
| 37 | + int shift; | |
| 38 | + target_phys_addr_t base; | |
| 39 | + qemu_irq intr; | |
| 40 | + qemu_irq rdy; | |
| 41 | + BlockDriverState *bdrv; | |
| 42 | + BlockDriverState *bdrv_cur; | |
| 43 | + uint8_t *image; | |
| 44 | + uint8_t *otp; | |
| 45 | + uint8_t *current; | |
| 46 | + ram_addr_t ram; | |
| 47 | + uint8_t *boot[2]; | |
| 48 | + uint8_t *data[2][2]; | |
| 49 | + int iomemtype; | |
| 50 | + int cycle; | |
| 51 | + int otpmode; | |
| 52 | + | |
| 53 | + uint16_t addr[8]; | |
| 54 | + uint16_t unladdr[8]; | |
| 55 | + int bufaddr; | |
| 56 | + int count; | |
| 57 | + uint16_t command; | |
| 58 | + uint16_t config[2]; | |
| 59 | + uint16_t status; | |
| 60 | + uint16_t intstatus; | |
| 61 | + uint16_t wpstatus; | |
| 62 | + | |
| 63 | + struct ecc_state_s ecc; | |
| 64 | + | |
| 65 | + int density_mask; | |
| 66 | + int secs; | |
| 67 | + int secs_cur; | |
| 68 | + int blocks; | |
| 69 | + uint8_t *blockwp; | |
| 70 | +}; | |
| 71 | + | |
| 72 | +enum { | |
| 73 | + ONEN_BUF_BLOCK = 0, | |
| 74 | + ONEN_BUF_BLOCK2 = 1, | |
| 75 | + ONEN_BUF_DEST_BLOCK = 2, | |
| 76 | + ONEN_BUF_DEST_PAGE = 3, | |
| 77 | + ONEN_BUF_PAGE = 7, | |
| 78 | +}; | |
| 79 | + | |
| 80 | +enum { | |
| 81 | + ONEN_ERR_CMD = 1 << 10, | |
| 82 | + ONEN_ERR_ERASE = 1 << 11, | |
| 83 | + ONEN_ERR_PROG = 1 << 12, | |
| 84 | + ONEN_ERR_LOAD = 1 << 13, | |
| 85 | +}; | |
| 86 | + | |
| 87 | +enum { | |
| 88 | + ONEN_INT_RESET = 1 << 4, | |
| 89 | + ONEN_INT_ERASE = 1 << 5, | |
| 90 | + ONEN_INT_PROG = 1 << 6, | |
| 91 | + ONEN_INT_LOAD = 1 << 7, | |
| 92 | + ONEN_INT = 1 << 15, | |
| 93 | +}; | |
| 94 | + | |
| 95 | +enum { | |
| 96 | + ONEN_LOCK_LOCKTIGHTEN = 1 << 0, | |
| 97 | + ONEN_LOCK_LOCKED = 1 << 1, | |
| 98 | + ONEN_LOCK_UNLOCKED = 1 << 2, | |
| 99 | +}; | |
| 100 | + | |
| 101 | +void onenand_base_update(void *opaque, target_phys_addr_t new) | |
| 102 | +{ | |
| 103 | + struct onenand_s *s = (struct onenand_s *) opaque; | |
| 104 | + | |
| 105 | + s->base = new; | |
| 106 | + | |
| 107 | + /* XXX: We should use IO_MEM_ROMD but we broke it earlier... | |
| 108 | + * Both 0x0000 ... 0x01ff and 0x8000 ... 0x800f can be used to | |
| 109 | + * write boot commands. Also take note of the BWPS bit. */ | |
| 110 | + cpu_register_physical_memory(s->base + (0x0000 << s->shift), | |
| 111 | + 0x0200 << s->shift, s->iomemtype); | |
| 112 | + cpu_register_physical_memory(s->base + (0x0200 << s->shift), | |
| 113 | + 0xbe00 << s->shift, | |
| 114 | + (s->ram +(0x0200 << s->shift)) | IO_MEM_RAM); | |
| 115 | + if (s->iomemtype) | |
| 116 | + cpu_register_physical_memory(s->base + (0xc000 << s->shift), | |
| 117 | + 0x4000 << s->shift, s->iomemtype); | |
| 118 | +} | |
| 119 | + | |
| 120 | +void onenand_base_unmap(void *opaque) | |
| 121 | +{ | |
| 122 | + struct onenand_s *s = (struct onenand_s *) opaque; | |
| 123 | + | |
| 124 | + cpu_register_physical_memory(s->base, | |
| 125 | + 0x10000 << s->shift, IO_MEM_UNASSIGNED); | |
| 126 | +} | |
| 127 | + | |
| 128 | +static void onenand_intr_update(struct onenand_s *s) | |
| 129 | +{ | |
| 130 | + qemu_set_irq(s->intr, ((s->intstatus >> 15) ^ (~s->config[0] >> 6)) & 1); | |
| 131 | +} | |
| 132 | + | |
| 133 | +/* Hot reset (Reset OneNAND command) or warm reset (RP pin low) */ | |
| 134 | +static void onenand_reset(struct onenand_s *s, int cold) | |
| 135 | +{ | |
| 136 | + memset(&s->addr, 0, sizeof(s->addr)); | |
| 137 | + s->command = 0; | |
| 138 | + s->count = 1; | |
| 139 | + s->bufaddr = 0; | |
| 140 | + s->config[0] = 0x40c0; | |
| 141 | + s->config[1] = 0x0000; | |
| 142 | + onenand_intr_update(s); | |
| 143 | + qemu_irq_raise(s->rdy); | |
| 144 | + s->status = 0x0000; | |
| 145 | + s->intstatus = cold ? 0x8080 : 0x8010; | |
| 146 | + s->unladdr[0] = 0; | |
| 147 | + s->unladdr[1] = 0; | |
| 148 | + s->wpstatus = 0x0002; | |
| 149 | + s->cycle = 0; | |
| 150 | + s->otpmode = 0; | |
| 151 | + s->bdrv_cur = s->bdrv; | |
| 152 | + s->current = s->image; | |
| 153 | + s->secs_cur = s->secs; | |
| 154 | + | |
| 155 | + if (cold) { | |
| 156 | + /* Lock the whole flash */ | |
| 157 | + memset(s->blockwp, ONEN_LOCK_LOCKED, s->blocks); | |
| 158 | + | |
| 159 | + if (s->bdrv && bdrv_read(s->bdrv, 0, s->boot[0], 8) < 0) | |
| 160 | + cpu_abort(cpu_single_env, "%s: Loading the BootRAM failed.\n", | |
| 161 | + __FUNCTION__); | |
| 162 | + } | |
| 163 | +} | |
| 164 | + | |
| 165 | +static inline int onenand_load_main(struct onenand_s *s, int sec, int secn, | |
| 166 | + void *dest) | |
| 167 | +{ | |
| 168 | + if (s->bdrv_cur) | |
| 169 | + return bdrv_read(s->bdrv_cur, sec, dest, secn) < 0; | |
| 170 | + else if (sec + secn > s->secs_cur) | |
| 171 | + return 1; | |
| 172 | + | |
| 173 | + memcpy(dest, s->current + (sec << 9), secn << 9); | |
| 174 | + | |
| 175 | + return 0; | |
| 176 | +} | |
| 177 | + | |
| 178 | +static inline int onenand_prog_main(struct onenand_s *s, int sec, int secn, | |
| 179 | + void *src) | |
| 180 | +{ | |
| 181 | + if (s->bdrv_cur) | |
| 182 | + return bdrv_write(s->bdrv_cur, sec, src, secn) < 0; | |
| 183 | + else if (sec + secn > s->secs_cur) | |
| 184 | + return 1; | |
| 185 | + | |
| 186 | + memcpy(s->current + (sec << 9), src, secn << 9); | |
| 187 | + | |
| 188 | + return 0; | |
| 189 | +} | |
| 190 | + | |
| 191 | +static inline int onenand_load_spare(struct onenand_s *s, int sec, int secn, | |
| 192 | + void *dest) | |
| 193 | +{ | |
| 194 | + uint8_t buf[512]; | |
| 195 | + | |
| 196 | + if (s->bdrv_cur) { | |
| 197 | + if (bdrv_read(s->bdrv_cur, s->secs_cur + (sec >> 5), buf, 1) < 0) | |
| 198 | + return 1; | |
| 199 | + memcpy(dest, buf + ((sec & 31) << 4), secn << 4); | |
| 200 | + } else if (sec + secn > s->secs_cur) | |
| 201 | + return 1; | |
| 202 | + else | |
| 203 | + memcpy(dest, s->current + (s->secs_cur << 9) + (sec << 4), secn << 4); | |
| 204 | + | |
| 205 | + return 0; | |
| 206 | +} | |
| 207 | + | |
| 208 | +static inline int onenand_prog_spare(struct onenand_s *s, int sec, int secn, | |
| 209 | + void *src) | |
| 210 | +{ | |
| 211 | + uint8_t buf[512]; | |
| 212 | + | |
| 213 | + if (s->bdrv_cur) { | |
| 214 | + if (bdrv_read(s->bdrv_cur, s->secs_cur + (sec >> 5), buf, 1) < 0) | |
| 215 | + return 1; | |
| 216 | + memcpy(buf + ((sec & 31) << 4), src, secn << 4); | |
| 217 | + return bdrv_write(s->bdrv_cur, s->secs_cur + (sec >> 5), buf, 1) < 0; | |
| 218 | + } else if (sec + secn > s->secs_cur) | |
| 219 | + return 1; | |
| 220 | + | |
| 221 | + memcpy(s->current + (s->secs_cur << 9) + (sec << 4), src, secn << 4); | |
| 222 | + | |
| 223 | + return 0; | |
| 224 | +} | |
| 225 | + | |
| 226 | +static inline int onenand_erase(struct onenand_s *s, int sec, int num) | |
| 227 | +{ | |
| 228 | + /* TODO: optimise */ | |
| 229 | + uint8_t buf[512]; | |
| 230 | + | |
| 231 | + memset(buf, 0xff, sizeof(buf)); | |
| 232 | + for (; num > 0; num --, sec ++) { | |
| 233 | + if (onenand_prog_main(s, sec, 1, buf)) | |
| 234 | + return 1; | |
| 235 | + if (onenand_prog_spare(s, sec, 1, buf)) | |
| 236 | + return 1; | |
| 237 | + } | |
| 238 | + | |
| 239 | + return 0; | |
| 240 | +} | |
| 241 | + | |
| 242 | +static void onenand_command(struct onenand_s *s, int cmd) | |
| 243 | +{ | |
| 244 | + int b; | |
| 245 | + int sec; | |
| 246 | + void *buf; | |
| 247 | +#define SETADDR(block, page) \ | |
| 248 | + sec = (s->addr[page] & 3) + \ | |
| 249 | + ((((s->addr[page] >> 2) & 0x3f) + \ | |
| 250 | + (((s->addr[block] & 0xfff) | \ | |
| 251 | + (s->addr[block] >> 15 ? \ | |
| 252 | + s->density_mask : 0)) << 6)) << (PAGE_SHIFT - 9)); | |
| 253 | +#define SETBUF_M() \ | |
| 254 | + buf = (s->bufaddr & 8) ? \ | |
| 255 | + s->data[(s->bufaddr >> 2) & 1][0] : s->boot[0]; \ | |
| 256 | + buf += (s->bufaddr & 3) << 9; | |
| 257 | +#define SETBUF_S() \ | |
| 258 | + buf = (s->bufaddr & 8) ? \ | |
| 259 | + s->data[(s->bufaddr >> 2) & 1][1] : s->boot[1]; \ | |
| 260 | + buf += (s->bufaddr & 3) << 4; | |
| 261 | + | |
| 262 | + switch (cmd) { | |
| 263 | + case 0x00: /* Load single/multiple sector data unit into buffer */ | |
| 264 | + SETADDR(ONEN_BUF_BLOCK, ONEN_BUF_PAGE) | |
| 265 | + | |
| 266 | + SETBUF_M() | |
| 267 | + if (onenand_load_main(s, sec, s->count, buf)) | |
| 268 | + s->status |= ONEN_ERR_CMD | ONEN_ERR_LOAD; | |
| 269 | + | |
| 270 | +#if 0 | |
| 271 | + SETBUF_S() | |
| 272 | + if (onenand_load_spare(s, sec, s->count, buf)) | |
| 273 | + s->status |= ONEN_ERR_CMD | ONEN_ERR_LOAD; | |
| 274 | +#endif | |
| 275 | + | |
| 276 | + /* TODO: if (s->bufaddr & 3) + s->count was > 4 (2k-pages) | |
| 277 | + * or if (s->bufaddr & 1) + s->count was > 2 (1k-pages) | |
| 278 | + * then we need two split the read/write into two chunks. | |
| 279 | + */ | |
| 280 | + s->intstatus |= ONEN_INT | ONEN_INT_LOAD; | |
| 281 | + break; | |
| 282 | + case 0x13: /* Load single/multiple spare sector into buffer */ | |
| 283 | + SETADDR(ONEN_BUF_BLOCK, ONEN_BUF_PAGE) | |
| 284 | + | |
| 285 | + SETBUF_S() | |
| 286 | + if (onenand_load_spare(s, sec, s->count, buf)) | |
| 287 | + s->status |= ONEN_ERR_CMD | ONEN_ERR_LOAD; | |
| 288 | + | |
| 289 | + /* TODO: if (s->bufaddr & 3) + s->count was > 4 (2k-pages) | |
| 290 | + * or if (s->bufaddr & 1) + s->count was > 2 (1k-pages) | |
| 291 | + * then we need two split the read/write into two chunks. | |
| 292 | + */ | |
| 293 | + s->intstatus |= ONEN_INT | ONEN_INT_LOAD; | |
| 294 | + break; | |
| 295 | + case 0x80: /* Program single/multiple sector data unit from buffer */ | |
| 296 | + SETADDR(ONEN_BUF_BLOCK, ONEN_BUF_PAGE) | |
| 297 | + | |
| 298 | + SETBUF_M() | |
| 299 | + if (onenand_prog_main(s, sec, s->count, buf)) | |
| 300 | + s->status |= ONEN_ERR_CMD | ONEN_ERR_PROG; | |
| 301 | + | |
| 302 | +#if 0 | |
| 303 | + SETBUF_S() | |
| 304 | + if (onenand_prog_spare(s, sec, s->count, buf)) | |
| 305 | + s->status |= ONEN_ERR_CMD | ONEN_ERR_PROG; | |
| 306 | +#endif | |
| 307 | + | |
| 308 | + /* TODO: if (s->bufaddr & 3) + s->count was > 4 (2k-pages) | |
| 309 | + * or if (s->bufaddr & 1) + s->count was > 2 (1k-pages) | |
| 310 | + * then we need two split the read/write into two chunks. | |
| 311 | + */ | |
| 312 | + s->intstatus |= ONEN_INT | ONEN_INT_PROG; | |
| 313 | + break; | |
| 314 | + case 0x1a: /* Program single/multiple spare area sector from buffer */ | |
| 315 | + SETADDR(ONEN_BUF_BLOCK, ONEN_BUF_PAGE) | |
| 316 | + | |
| 317 | + SETBUF_S() | |
| 318 | + if (onenand_prog_spare(s, sec, s->count, buf)) | |
| 319 | + s->status |= ONEN_ERR_CMD | ONEN_ERR_PROG; | |
| 320 | + | |
| 321 | + /* TODO: if (s->bufaddr & 3) + s->count was > 4 (2k-pages) | |
| 322 | + * or if (s->bufaddr & 1) + s->count was > 2 (1k-pages) | |
| 323 | + * then we need two split the read/write into two chunks. | |
| 324 | + */ | |
| 325 | + s->intstatus |= ONEN_INT | ONEN_INT_PROG; | |
| 326 | + break; | |
| 327 | + case 0x1b: /* Copy-back program */ | |
| 328 | + SETBUF_S() | |
| 329 | + | |
| 330 | + SETADDR(ONEN_BUF_BLOCK, ONEN_BUF_PAGE) | |
| 331 | + if (onenand_load_main(s, sec, s->count, buf)) | |
| 332 | + s->status |= ONEN_ERR_CMD | ONEN_ERR_PROG; | |
| 333 | + | |
| 334 | + SETADDR(ONEN_BUF_DEST_BLOCK, ONEN_BUF_DEST_PAGE) | |
| 335 | + if (onenand_prog_main(s, sec, s->count, buf)) | |
| 336 | + s->status |= ONEN_ERR_CMD | ONEN_ERR_PROG; | |
| 337 | + | |
| 338 | + /* TODO: spare areas */ | |
| 339 | + | |
| 340 | + s->intstatus |= ONEN_INT | ONEN_INT_PROG; | |
| 341 | + break; | |
| 342 | + | |
| 343 | + case 0x23: /* Unlock NAND array block(s) */ | |
| 344 | + s->intstatus |= ONEN_INT; | |
| 345 | + | |
| 346 | + /* XXX the previous (?) area should be locked automatically */ | |
| 347 | + for (b = s->unladdr[0]; b <= s->unladdr[1]; b ++) { | |
| 348 | + if (b >= s->blocks) { | |
| 349 | + s->status |= ONEN_ERR_CMD; | |
| 350 | + break; | |
| 351 | + } | |
| 352 | + if (s->blockwp[b] == ONEN_LOCK_LOCKTIGHTEN) | |
| 353 | + break; | |
| 354 | + | |
| 355 | + s->wpstatus = s->blockwp[b] = ONEN_LOCK_UNLOCKED; | |
| 356 | + } | |
| 357 | + break; | |
| 358 | + case 0x2a: /* Lock NAND array block(s) */ | |
| 359 | + s->intstatus |= ONEN_INT; | |
| 360 | + | |
| 361 | + for (b = s->unladdr[0]; b <= s->unladdr[1]; b ++) { | |
| 362 | + if (b >= s->blocks) { | |
| 363 | + s->status |= ONEN_ERR_CMD; | |
| 364 | + break; | |
| 365 | + } | |
| 366 | + if (s->blockwp[b] == ONEN_LOCK_LOCKTIGHTEN) | |
| 367 | + break; | |
| 368 | + | |
| 369 | + s->wpstatus = s->blockwp[b] = ONEN_LOCK_LOCKED; | |
| 370 | + } | |
| 371 | + break; | |
| 372 | + case 0x2c: /* Lock-tight NAND array block(s) */ | |
| 373 | + s->intstatus |= ONEN_INT; | |
| 374 | + | |
| 375 | + for (b = s->unladdr[0]; b <= s->unladdr[1]; b ++) { | |
| 376 | + if (b >= s->blocks) { | |
| 377 | + s->status |= ONEN_ERR_CMD; | |
| 378 | + break; | |
| 379 | + } | |
| 380 | + if (s->blockwp[b] == ONEN_LOCK_UNLOCKED) | |
| 381 | + continue; | |
| 382 | + | |
| 383 | + s->wpstatus = s->blockwp[b] = ONEN_LOCK_LOCKTIGHTEN; | |
| 384 | + } | |
| 385 | + break; | |
| 386 | + | |
| 387 | + case 0x71: /* Erase-Verify-Read */ | |
| 388 | + s->intstatus |= ONEN_INT; | |
| 389 | + break; | |
| 390 | + case 0x95: /* Multi-block erase */ | |
| 391 | + qemu_irq_pulse(s->intr); | |
| 392 | + /* Fall through. */ | |
| 393 | + case 0x94: /* Block erase */ | |
| 394 | + sec = ((s->addr[ONEN_BUF_BLOCK] & 0xfff) | | |
| 395 | + (s->addr[ONEN_BUF_BLOCK] >> 15 ? s->density_mask : 0)) | |
| 396 | + << (BLOCK_SHIFT - 9); | |
| 397 | + if (onenand_erase(s, sec, 1 << (BLOCK_SHIFT - 9))) | |
| 398 | + s->status |= ONEN_ERR_CMD | ONEN_ERR_ERASE; | |
| 399 | + | |
| 400 | + s->intstatus |= ONEN_INT | ONEN_INT_ERASE; | |
| 401 | + break; | |
| 402 | + case 0xb0: /* Erase suspend */ | |
| 403 | + break; | |
| 404 | + case 0x30: /* Erase resume */ | |
| 405 | + s->intstatus |= ONEN_INT | ONEN_INT_ERASE; | |
| 406 | + break; | |
| 407 | + | |
| 408 | + case 0xf0: /* Reset NAND Flash core */ | |
| 409 | + onenand_reset(s, 0); | |
| 410 | + break; | |
| 411 | + case 0xf3: /* Reset OneNAND */ | |
| 412 | + onenand_reset(s, 0); | |
| 413 | + break; | |
| 414 | + | |
| 415 | + case 0x65: /* OTP Access */ | |
| 416 | + s->intstatus |= ONEN_INT; | |
| 417 | + s->bdrv_cur = 0; | |
| 418 | + s->current = s->otp; | |
| 419 | + s->secs_cur = 1 << (BLOCK_SHIFT - 9); | |
| 420 | + s->addr[ONEN_BUF_BLOCK] = 0; | |
| 421 | + s->otpmode = 1; | |
| 422 | + break; | |
| 423 | + | |
| 424 | + default: | |
| 425 | + s->status |= ONEN_ERR_CMD; | |
| 426 | + s->intstatus |= ONEN_INT; | |
| 427 | + fprintf(stderr, "%s: unknown OneNAND command %x\n", | |
| 428 | + __FUNCTION__, cmd); | |
| 429 | + } | |
| 430 | + | |
| 431 | + onenand_intr_update(s); | |
| 432 | +} | |
| 433 | + | |
| 434 | +static uint32_t onenand_read(void *opaque, target_phys_addr_t addr) | |
| 435 | +{ | |
| 436 | + struct onenand_s *s = (struct onenand_s *) opaque; | |
| 437 | + int offset = (addr - s->base) >> s->shift; | |
| 438 | + | |
| 439 | + switch (offset) { | |
| 440 | + case 0x0000 ... 0xc000: | |
| 441 | + return lduw_le_p(s->boot[0] + (addr - s->base)); | |
| 442 | + | |
| 443 | + case 0xf000: /* Manufacturer ID */ | |
| 444 | + return (s->id >> 16) & 0xff; | |
| 445 | + case 0xf001: /* Device ID */ | |
| 446 | + return (s->id >> 8) & 0xff; | |
| 447 | + /* TODO: get the following values from a real chip! */ | |
| 448 | + case 0xf002: /* Version ID */ | |
| 449 | + return (s->id >> 0) & 0xff; | |
| 450 | + case 0xf003: /* Data Buffer size */ | |
| 451 | + return 1 << PAGE_SHIFT; | |
| 452 | + case 0xf004: /* Boot Buffer size */ | |
| 453 | + return 0x200; | |
| 454 | + case 0xf005: /* Amount of buffers */ | |
| 455 | + return 1 | (2 << 8); | |
| 456 | + case 0xf006: /* Technology */ | |
| 457 | + return 0; | |
| 458 | + | |
| 459 | + case 0xf100 ... 0xf107: /* Start addresses */ | |
| 460 | + return s->addr[offset - 0xf100]; | |
| 461 | + | |
| 462 | + case 0xf200: /* Start buffer */ | |
| 463 | + return (s->bufaddr << 8) | ((s->count - 1) & (1 << (PAGE_SHIFT - 10))); | |
| 464 | + | |
| 465 | + case 0xf220: /* Command */ | |
| 466 | + return s->command; | |
| 467 | + case 0xf221: /* System Configuration 1 */ | |
| 468 | + return s->config[0] & 0xffe0; | |
| 469 | + case 0xf222: /* System Configuration 2 */ | |
| 470 | + return s->config[1]; | |
| 471 | + | |
| 472 | + case 0xf240: /* Controller Status */ | |
| 473 | + return s->status; | |
| 474 | + case 0xf241: /* Interrupt */ | |
| 475 | + return s->intstatus; | |
| 476 | + case 0xf24c: /* Unlock Start Block Address */ | |
| 477 | + return s->unladdr[0]; | |
| 478 | + case 0xf24d: /* Unlock End Block Address */ | |
| 479 | + return s->unladdr[1]; | |
| 480 | + case 0xf24e: /* Write Protection Status */ | |
| 481 | + return s->wpstatus; | |
| 482 | + | |
| 483 | + case 0xff00: /* ECC Status */ | |
| 484 | + return 0x00; | |
| 485 | + case 0xff01: /* ECC Result of main area data */ | |
| 486 | + case 0xff02: /* ECC Result of spare area data */ | |
| 487 | + case 0xff03: /* ECC Result of main area data */ | |
| 488 | + case 0xff04: /* ECC Result of spare area data */ | |
| 489 | + cpu_abort(cpu_single_env, "%s: imeplement ECC\n", __FUNCTION__); | |
| 490 | + return 0x0000; | |
| 491 | + } | |
| 492 | + | |
| 493 | + fprintf(stderr, "%s: unknown OneNAND register %x\n", | |
| 494 | + __FUNCTION__, offset); | |
| 495 | + return 0; | |
| 496 | +} | |
| 497 | + | |
| 498 | +static void onenand_write(void *opaque, target_phys_addr_t addr, | |
| 499 | + uint32_t value) | |
| 500 | +{ | |
| 501 | + struct onenand_s *s = (struct onenand_s *) opaque; | |
| 502 | + int offset = (addr - s->base) >> s->shift; | |
| 503 | + int sec; | |
| 504 | + | |
| 505 | + switch (offset) { | |
| 506 | + case 0x0000 ... 0x01ff: | |
| 507 | + case 0x8000 ... 0x800f: | |
| 508 | + if (s->cycle) { | |
| 509 | + s->cycle = 0; | |
| 510 | + | |
| 511 | + if (value == 0x0000) { | |
| 512 | + SETADDR(ONEN_BUF_BLOCK, ONEN_BUF_PAGE) | |
| 513 | + onenand_load_main(s, sec, | |
| 514 | + 1 << (PAGE_SHIFT - 9), s->data[0][0]); | |
| 515 | + s->addr[ONEN_BUF_PAGE] += 4; | |
| 516 | + s->addr[ONEN_BUF_PAGE] &= 0xff; | |
| 517 | + } | |
| 518 | + break; | |
| 519 | + } | |
| 520 | + | |
| 521 | + switch (value) { | |
| 522 | + case 0x00f0: /* Reset OneNAND */ | |
| 523 | + onenand_reset(s, 0); | |
| 524 | + break; | |
| 525 | + | |
| 526 | + case 0x00e0: /* Load Data into Buffer */ | |
| 527 | + s->cycle = 1; | |
| 528 | + break; | |
| 529 | + | |
| 530 | + case 0x0090: /* Read Identification Data */ | |
| 531 | + memset(s->boot[0], 0, 3 << s->shift); | |
| 532 | + s->boot[0][0 << s->shift] = (s->id >> 16) & 0xff; | |
| 533 | + s->boot[0][1 << s->shift] = (s->id >> 8) & 0xff; | |
| 534 | + s->boot[0][2 << s->shift] = s->wpstatus & 0xff; | |
| 535 | + break; | |
| 536 | + | |
| 537 | + default: | |
| 538 | + fprintf(stderr, "%s: unknown OneNAND boot command %x\n", | |
| 539 | + __FUNCTION__, value); | |
| 540 | + } | |
| 541 | + break; | |
| 542 | + | |
| 543 | + case 0xf100 ... 0xf107: /* Start addresses */ | |
| 544 | + s->addr[offset - 0xf100] = value; | |
| 545 | + break; | |
| 546 | + | |
| 547 | + case 0xf200: /* Start buffer */ | |
| 548 | + s->bufaddr = (value >> 8) & 0xf; | |
| 549 | + if (PAGE_SHIFT == 11) | |
| 550 | + s->count = (value & 3) ?: 4; | |
| 551 | + else if (PAGE_SHIFT == 10) | |
| 552 | + s->count = (value & 1) ?: 2; | |
| 553 | + break; | |
| 554 | + | |
| 555 | + case 0xf220: /* Command */ | |
| 556 | + if (s->intstatus & (1 << 15)) | |
| 557 | + break; | |
| 558 | + s->command = value; | |
| 559 | + onenand_command(s, s->command); | |
| 560 | + break; | |
| 561 | + case 0xf221: /* System Configuration 1 */ | |
| 562 | + s->config[0] = value; | |
| 563 | + onenand_intr_update(s); | |
| 564 | + qemu_set_irq(s->rdy, (s->config[0] >> 7) & 1); | |
| 565 | + break; | |
| 566 | + case 0xf222: /* System Configuration 2 */ | |
| 567 | + s->config[1] = value; | |
| 568 | + break; | |
| 569 | + | |
| 570 | + case 0xf241: /* Interrupt */ | |
| 571 | + s->intstatus &= value; | |
| 572 | + if ((1 << 15) & ~s->intstatus) | |
| 573 | + s->status &= ~(ONEN_ERR_CMD | ONEN_ERR_ERASE | | |
| 574 | + ONEN_ERR_PROG | ONEN_ERR_LOAD); | |
| 575 | + onenand_intr_update(s); | |
| 576 | + break; | |
| 577 | + case 0xf24c: /* Unlock Start Block Address */ | |
| 578 | + s->unladdr[0] = value & (s->blocks - 1); | |
| 579 | + /* For some reason we have to set the end address to by default | |
| 580 | + * be same as start because the software forgets to write anything | |
| 581 | + * in there. */ | |
| 582 | + s->unladdr[1] = value & (s->blocks - 1); | |
| 583 | + break; | |
| 584 | + case 0xf24d: /* Unlock End Block Address */ | |
| 585 | + s->unladdr[1] = value & (s->blocks - 1); | |
| 586 | + break; | |
| 587 | + | |
| 588 | + default: | |
| 589 | + fprintf(stderr, "%s: unknown OneNAND register %x\n", | |
| 590 | + __FUNCTION__, offset); | |
| 591 | + } | |
| 592 | +} | |
| 593 | + | |
| 594 | +static CPUReadMemoryFunc *onenand_readfn[] = { | |
| 595 | + onenand_read, /* TODO */ | |
| 596 | + onenand_read, | |
| 597 | + onenand_read, | |
| 598 | +}; | |
| 599 | + | |
| 600 | +static CPUWriteMemoryFunc *onenand_writefn[] = { | |
| 601 | + onenand_write, /* TODO */ | |
| 602 | + onenand_write, | |
| 603 | + onenand_write, | |
| 604 | +}; | |
| 605 | + | |
| 606 | +void *onenand_init(uint32_t id, int regshift, qemu_irq irq) | |
| 607 | +{ | |
| 608 | + struct onenand_s *s = (struct onenand_s *) qemu_mallocz(sizeof(*s)); | |
| 609 | + int bdrv_index = drive_get_index(IF_MTD, 0, 0); | |
| 610 | + uint32_t size = 1 << (24 + ((id >> 12) & 7)); | |
| 611 | + void *ram; | |
| 612 | + | |
| 613 | + s->shift = regshift; | |
| 614 | + s->intr = irq; | |
| 615 | + s->rdy = 0; | |
| 616 | + s->id = id; | |
| 617 | + s->blocks = size >> BLOCK_SHIFT; | |
| 618 | + s->secs = size >> 9; | |
| 619 | + s->blockwp = qemu_malloc(s->blocks); | |
| 620 | + s->density_mask = (id & (1 << 11)) ? (1 << (6 + ((id >> 12) & 7))) : 0; | |
| 621 | + s->iomemtype = cpu_register_io_memory(0, onenand_readfn, | |
| 622 | + onenand_writefn, s); | |
| 623 | + if (bdrv_index == -1) | |
| 624 | + s->image = memset(qemu_malloc(size + (size >> 5)), | |
| 625 | + 0xff, size + (size >> 5)); | |
| 626 | + else | |
| 627 | + s->bdrv = drives_table[bdrv_index].bdrv; | |
| 628 | + s->otp = memset(qemu_malloc((64 + 2) << PAGE_SHIFT), | |
| 629 | + 0xff, (64 + 2) << PAGE_SHIFT); | |
| 630 | + s->ram = qemu_ram_alloc(0xc000 << s->shift); | |
| 631 | + ram = phys_ram_base + s->ram; | |
| 632 | + s->boot[0] = ram + (0x0000 << s->shift); | |
| 633 | + s->boot[1] = ram + (0x8000 << s->shift); | |
| 634 | + s->data[0][0] = ram + ((0x0200 + (0 << (PAGE_SHIFT - 1))) << s->shift); | |
| 635 | + s->data[0][1] = ram + ((0x8010 + (0 << (PAGE_SHIFT - 6))) << s->shift); | |
| 636 | + s->data[1][0] = ram + ((0x0200 + (1 << (PAGE_SHIFT - 1))) << s->shift); | |
| 637 | + s->data[1][1] = ram + ((0x8010 + (1 << (PAGE_SHIFT - 6))) << s->shift); | |
| 638 | + | |
| 639 | + onenand_reset(s, 1); | |
| 640 | + | |
| 641 | + return s; | |
| 642 | +} | ... | ... |
hw/tmp105.c
0 → 100644
| 1 | +/* | |
| 2 | + * Texas Instruments TMP105 temperature sensor. | |
| 3 | + * | |
| 4 | + * Copyright (C) 2008 Nokia Corporation | |
| 5 | + * Written by Andrzej Zaborowski <andrew@openedhand.com> | |
| 6 | + * | |
| 7 | + * This program is free software; you can redistribute it and/or | |
| 8 | + * modify it under the terms of the GNU General Public License as | |
| 9 | + * published by the Free Software Foundation; either version 2 or | |
| 10 | + * (at your option) version 3 of the License. | |
| 11 | + * | |
| 12 | + * This program is distributed in the hope that it will be useful, | |
| 13 | + * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
| 14 | + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
| 15 | + * GNU General Public License for more details. | |
| 16 | + * | |
| 17 | + * You should have received a copy of the GNU General Public License | |
| 18 | + * along with this program; if not, write to the Free Software | |
| 19 | + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, | |
| 20 | + * MA 02111-1307 USA | |
| 21 | + */ | |
| 22 | + | |
| 23 | +#include "hw.h" | |
| 24 | +#include "i2c.h" | |
| 25 | + | |
| 26 | +struct tmp105_s { | |
| 27 | + i2c_slave i2c; | |
| 28 | + int len; | |
| 29 | + uint8_t buf[2]; | |
| 30 | + qemu_irq pin; | |
| 31 | + | |
| 32 | + uint8_t pointer; | |
| 33 | + uint8_t config; | |
| 34 | + int16_t temperature; | |
| 35 | + int16_t limit[2]; | |
| 36 | + int faults; | |
| 37 | + int alarm; | |
| 38 | +}; | |
| 39 | + | |
| 40 | +static void tmp105_interrupt_update(struct tmp105_s *s) | |
| 41 | +{ | |
| 42 | + qemu_set_irq(s->pin, s->alarm ^ ((~s->config >> 2) & 1)); /* POL */ | |
| 43 | +} | |
| 44 | + | |
| 45 | +static void tmp105_alarm_update(struct tmp105_s *s) | |
| 46 | +{ | |
| 47 | + if ((s->config >> 0) & 1) { /* SD */ | |
| 48 | + if ((s->config >> 7) & 1) /* OS */ | |
| 49 | + s->config &= ~(1 << 7); /* OS */ | |
| 50 | + else | |
| 51 | + return; | |
| 52 | + } | |
| 53 | + | |
| 54 | + if ((s->config >> 1) & 1) { /* TM */ | |
| 55 | + if (s->temperature >= s->limit[1]) | |
| 56 | + s->alarm = 1; | |
| 57 | + else if (s->temperature < s->limit[0]) | |
| 58 | + s->alarm = 1; | |
| 59 | + } else { | |
| 60 | + if (s->temperature >= s->limit[1]) | |
| 61 | + s->alarm = 1; | |
| 62 | + else if (s->temperature < s->limit[0]) | |
| 63 | + s->alarm = 0; | |
| 64 | + } | |
| 65 | + | |
| 66 | + tmp105_interrupt_update(s); | |
| 67 | +} | |
| 68 | + | |
| 69 | +/* Units are 0.001 centigrades relative to 0 C. */ | |
| 70 | +void tmp105_set(i2c_slave *i2c, int temp) | |
| 71 | +{ | |
| 72 | + struct tmp105_s *s = (struct tmp105_s *) i2c; | |
| 73 | + | |
| 74 | + if (temp >= 128000 || temp < -128000) { | |
| 75 | + fprintf(stderr, "%s: values is out of range (%i.%03i C)\n", | |
| 76 | + __FUNCTION__, temp / 1000, temp % 1000); | |
| 77 | + exit(-1); | |
| 78 | + } | |
| 79 | + | |
| 80 | + s->temperature = ((int16_t) (temp * 0x800 / 128000)) << 4; | |
| 81 | + | |
| 82 | + tmp105_alarm_update(s); | |
| 83 | +} | |
| 84 | + | |
| 85 | +static const int tmp105_faultq[4] = { 1, 2, 4, 6 }; | |
| 86 | + | |
| 87 | +static void tmp105_read(struct tmp105_s *s) | |
| 88 | +{ | |
| 89 | + s->len = 0; | |
| 90 | + | |
| 91 | + if ((s->config >> 1) & 1) { /* TM */ | |
| 92 | + s->alarm = 0; | |
| 93 | + tmp105_interrupt_update(s); | |
| 94 | + } | |
| 95 | + | |
| 96 | + switch (s->pointer & 3) { | |
| 97 | + case 0: /* Temperature */ | |
| 98 | + s->buf[s->len ++] = (((uint16_t) s->temperature) >> 8); | |
| 99 | + s->buf[s->len ++] = (((uint16_t) s->temperature) >> 0) & | |
| 100 | + (0xf0 << ((~s->config >> 5) & 3)); /* R */ | |
| 101 | + break; | |
| 102 | + | |
| 103 | + case 1: /* Configuration */ | |
| 104 | + s->buf[s->len ++] = s->config; | |
| 105 | + break; | |
| 106 | + | |
| 107 | + case 2: /* T_LOW */ | |
| 108 | + s->buf[s->len ++] = ((uint16_t) s->limit[0]) >> 8; | |
| 109 | + s->buf[s->len ++] = ((uint16_t) s->limit[0]) >> 0; | |
| 110 | + break; | |
| 111 | + | |
| 112 | + case 3: /* T_HIGH */ | |
| 113 | + s->buf[s->len ++] = ((uint16_t) s->limit[1]) >> 8; | |
| 114 | + s->buf[s->len ++] = ((uint16_t) s->limit[1]) >> 0; | |
| 115 | + break; | |
| 116 | + } | |
| 117 | +} | |
| 118 | + | |
| 119 | +static void tmp105_write(struct tmp105_s *s) | |
| 120 | +{ | |
| 121 | + switch (s->pointer & 3) { | |
| 122 | + case 0: /* Temperature */ | |
| 123 | + break; | |
| 124 | + | |
| 125 | + case 1: /* Configuration */ | |
| 126 | + if (s->buf[0] & ~s->config & (1 << 0)) /* SD */ | |
| 127 | + printf("%s: TMP105 shutdown\n", __FUNCTION__); | |
| 128 | + s->config = s->buf[0]; | |
| 129 | + s->faults = tmp105_faultq[(s->config >> 3) & 3]; /* F */ | |
| 130 | + tmp105_alarm_update(s); | |
| 131 | + break; | |
| 132 | + | |
| 133 | + case 2: /* T_LOW */ | |
| 134 | + case 3: /* T_HIGH */ | |
| 135 | + if (s->len >= 3) | |
| 136 | + s->limit[s->pointer & 1] = (int16_t) | |
| 137 | + ((((uint16_t) s->buf[0]) << 8) | s->buf[1]); | |
| 138 | + tmp105_alarm_update(s); | |
| 139 | + break; | |
| 140 | + } | |
| 141 | +} | |
| 142 | + | |
| 143 | +static int tmp105_rx(i2c_slave *i2c) | |
| 144 | +{ | |
| 145 | + struct tmp105_s *s = (struct tmp105_s *) i2c; | |
| 146 | + | |
| 147 | + if (s->len < 2) | |
| 148 | + return s->buf[s->len ++]; | |
| 149 | + else | |
| 150 | + return 0xff; | |
| 151 | +} | |
| 152 | + | |
| 153 | +static int tmp105_tx(i2c_slave *i2c, uint8_t data) | |
| 154 | +{ | |
| 155 | + struct tmp105_s *s = (struct tmp105_s *) i2c; | |
| 156 | + | |
| 157 | + if (!s->len ++) | |
| 158 | + s->pointer = data; | |
| 159 | + else { | |
| 160 | + if (s->len <= 2) | |
| 161 | + s->buf[s->len - 1] = data; | |
| 162 | + tmp105_write(s); | |
| 163 | + } | |
| 164 | + | |
| 165 | + return 0; | |
| 166 | +} | |
| 167 | + | |
| 168 | +static void tmp105_event(i2c_slave *i2c, enum i2c_event event) | |
| 169 | +{ | |
| 170 | + struct tmp105_s *s = (struct tmp105_s *) i2c; | |
| 171 | + | |
| 172 | + if (event == I2C_START_RECV) | |
| 173 | + tmp105_read(s); | |
| 174 | + | |
| 175 | + s->len = 0; | |
| 176 | +} | |
| 177 | + | |
| 178 | +static void tmp105_save(QEMUFile *f, void *opaque) | |
| 179 | +{ | |
| 180 | + struct tmp105_s *s = (struct tmp105_s *) opaque; | |
| 181 | + | |
| 182 | + qemu_put_byte(f, s->len); | |
| 183 | + qemu_put_8s(f, &s->buf[0]); | |
| 184 | + qemu_put_8s(f, &s->buf[1]); | |
| 185 | + | |
| 186 | + qemu_put_8s(f, &s->pointer); | |
| 187 | + qemu_put_8s(f, &s->config); | |
| 188 | + qemu_put_be16s(f, &s->temperature); | |
| 189 | + qemu_put_be16s(f, &s->limit[0]); | |
| 190 | + qemu_put_be16s(f, &s->limit[1]); | |
| 191 | + qemu_put_byte(f, s->alarm); | |
| 192 | + s->faults = tmp105_faultq[(s->config >> 3) & 3]; /* F */ | |
| 193 | + | |
| 194 | + i2c_slave_save(f, &s->i2c); | |
| 195 | +} | |
| 196 | + | |
| 197 | +static int tmp105_load(QEMUFile *f, void *opaque, int version_id) | |
| 198 | +{ | |
| 199 | + struct tmp105_s *s = (struct tmp105_s *) opaque; | |
| 200 | + | |
| 201 | + s->len = qemu_get_byte(f); | |
| 202 | + qemu_get_8s(f, &s->buf[0]); | |
| 203 | + qemu_get_8s(f, &s->buf[1]); | |
| 204 | + | |
| 205 | + qemu_get_8s(f, &s->pointer); | |
| 206 | + qemu_get_8s(f, &s->config); | |
| 207 | + qemu_get_be16s(f, &s->temperature); | |
| 208 | + qemu_get_be16s(f, &s->limit[0]); | |
| 209 | + qemu_get_be16s(f, &s->limit[1]); | |
| 210 | + s->alarm = qemu_get_byte(f); | |
| 211 | + | |
| 212 | + tmp105_interrupt_update(s); | |
| 213 | + | |
| 214 | + i2c_slave_load(f, &s->i2c); | |
| 215 | + return 0; | |
| 216 | +} | |
| 217 | + | |
| 218 | +void tmp105_reset(i2c_slave *i2c) | |
| 219 | +{ | |
| 220 | + struct tmp105_s *s = (struct tmp105_s *) i2c; | |
| 221 | + | |
| 222 | + s->temperature = 0; | |
| 223 | + s->pointer = 0; | |
| 224 | + s->config = 0; | |
| 225 | + s->faults = tmp105_faultq[(s->config >> 3) & 3]; | |
| 226 | + s->alarm = 0; | |
| 227 | + | |
| 228 | + tmp105_interrupt_update(s); | |
| 229 | +} | |
| 230 | + | |
| 231 | +static int tmp105_iid = 0; | |
| 232 | + | |
| 233 | +struct i2c_slave *tmp105_init(i2c_bus *bus, qemu_irq alarm) | |
| 234 | +{ | |
| 235 | + struct tmp105_s *s = (struct tmp105_s *) | |
| 236 | + i2c_slave_init(bus, 0, sizeof(struct tmp105_s)); | |
| 237 | + | |
| 238 | + s->i2c.event = tmp105_event; | |
| 239 | + s->i2c.recv = tmp105_rx; | |
| 240 | + s->i2c.send = tmp105_tx; | |
| 241 | + s->pin = alarm; | |
| 242 | + | |
| 243 | + tmp105_reset(&s->i2c); | |
| 244 | + | |
| 245 | + register_savevm("TMP105", tmp105_iid ++, 0, | |
| 246 | + tmp105_save, tmp105_load, s); | |
| 247 | + | |
| 248 | + return &s->i2c; | |
| 249 | +} | ... | ... |
hw/twl92230.c
0 → 100644
| 1 | +/* | |
| 2 | + * TI TWL92230C energy-management companion device for the OMAP24xx. | |
| 3 | + * Aka. Menelaus (N4200 MENELAUS1_V2.2) | |
| 4 | + * | |
| 5 | + * Copyright (C) 2008 Nokia Corporation | |
| 6 | + * Written by Andrzej Zaborowski <andrew@openedhand.com> | |
| 7 | + * | |
| 8 | + * This program is free software; you can redistribute it and/or | |
| 9 | + * modify it under the terms of the GNU General Public License as | |
| 10 | + * published by the Free Software Foundation; either version 2 or | |
| 11 | + * (at your option) version 3 of the License. | |
| 12 | + * | |
| 13 | + * This program is distributed in the hope that it will be useful, | |
| 14 | + * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
| 15 | + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
| 16 | + * GNU General Public License for more details. | |
| 17 | + * | |
| 18 | + * You should have received a copy of the GNU General Public License | |
| 19 | + * along with this program; if not, write to the Free Software | |
| 20 | + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, | |
| 21 | + * MA 02111-1307 USA | |
| 22 | + */ | |
| 23 | + | |
| 24 | +#include "hw.h" | |
| 25 | +#include "qemu-timer.h" | |
| 26 | +#include "i2c.h" | |
| 27 | +#include "sysemu.h" | |
| 28 | +#include "console.h" | |
| 29 | + | |
| 30 | +#define VERBOSE 1 | |
| 31 | + | |
| 32 | +struct menelaus_s { | |
| 33 | + i2c_slave i2c; | |
| 34 | + qemu_irq irq; | |
| 35 | + | |
| 36 | + int firstbyte; | |
| 37 | + uint8_t reg; | |
| 38 | + | |
| 39 | + uint8_t vcore[5]; | |
| 40 | + uint8_t dcdc[3]; | |
| 41 | + uint8_t ldo[8]; | |
| 42 | + uint8_t sleep[2]; | |
| 43 | + uint8_t osc; | |
| 44 | + uint8_t detect; | |
| 45 | + uint16_t mask; | |
| 46 | + uint16_t status; | |
| 47 | + uint8_t dir; | |
| 48 | + uint8_t inputs; | |
| 49 | + uint8_t outputs; | |
| 50 | + uint8_t bbsms; | |
| 51 | + uint8_t pull[4]; | |
| 52 | + uint8_t mmc_ctrl[3]; | |
| 53 | + uint8_t mmc_debounce; | |
| 54 | + struct { | |
| 55 | + uint8_t ctrl; | |
| 56 | + uint16_t comp; | |
| 57 | + QEMUTimer *hz; | |
| 58 | + int64_t next; | |
| 59 | + struct tm tm; | |
| 60 | + struct tm new; | |
| 61 | + struct tm alm; | |
| 62 | + time_t sec; | |
| 63 | + time_t alm_sec; | |
| 64 | + time_t next_comp; | |
| 65 | + struct tm *(*gettime)(const time_t *timep, struct tm *result); | |
| 66 | + } rtc; | |
| 67 | + qemu_irq handler[3]; | |
| 68 | + qemu_irq *in; | |
| 69 | + int pwrbtn_state; | |
| 70 | + qemu_irq pwrbtn; | |
| 71 | +}; | |
| 72 | + | |
| 73 | +static inline void menelaus_update(struct menelaus_s *s) | |
| 74 | +{ | |
| 75 | + qemu_set_irq(s->irq, s->status & ~s->mask); | |
| 76 | +} | |
| 77 | + | |
| 78 | +static inline void menelaus_rtc_start(struct menelaus_s *s) | |
| 79 | +{ | |
| 80 | + s->rtc.next =+ qemu_get_clock(rt_clock); | |
| 81 | + qemu_mod_timer(s->rtc.hz, s->rtc.next); | |
| 82 | +} | |
| 83 | + | |
| 84 | +static inline void menelaus_rtc_stop(struct menelaus_s *s) | |
| 85 | +{ | |
| 86 | + qemu_del_timer(s->rtc.hz); | |
| 87 | + s->rtc.next =- qemu_get_clock(rt_clock); | |
| 88 | + if (s->rtc.next < 1) | |
| 89 | + s->rtc.next = 1; | |
| 90 | +} | |
| 91 | + | |
| 92 | +static void menelaus_rtc_update(struct menelaus_s *s) | |
| 93 | +{ | |
| 94 | + s->rtc.gettime(&s->rtc.sec, &s->rtc.tm); | |
| 95 | +} | |
| 96 | + | |
| 97 | +static void menelaus_alm_update(struct menelaus_s *s) | |
| 98 | +{ | |
| 99 | + if ((s->rtc.ctrl & 3) == 3) | |
| 100 | + s->rtc.alm_sec = mktime(&s->rtc.alm); | |
| 101 | +} | |
| 102 | + | |
| 103 | +static void menelaus_rtc_hz(void *opaque) | |
| 104 | +{ | |
| 105 | + struct menelaus_s *s = (struct menelaus_s *) opaque; | |
| 106 | + | |
| 107 | + s->rtc.sec ++; | |
| 108 | + s->rtc.next += 1000; | |
| 109 | + qemu_mod_timer(s->rtc.hz, s->rtc.next); | |
| 110 | + if ((s->rtc.ctrl >> 3) & 3) { /* EVERY */ | |
| 111 | + menelaus_rtc_update(s); | |
| 112 | + if (((s->rtc.ctrl >> 3) & 3) == 1 && !s->rtc.tm.tm_sec) | |
| 113 | + s->status |= 1 << 8; /* RTCTMR */ | |
| 114 | + else if (((s->rtc.ctrl >> 3) & 3) == 2 && !s->rtc.tm.tm_min) | |
| 115 | + s->status |= 1 << 8; /* RTCTMR */ | |
| 116 | + else if (!s->rtc.tm.tm_hour) | |
| 117 | + s->status |= 1 << 8; /* RTCTMR */ | |
| 118 | + } else | |
| 119 | + s->status |= 1 << 8; /* RTCTMR */ | |
| 120 | + if ((s->rtc.ctrl >> 1) & 1) { /* RTC_AL_EN */ | |
| 121 | + if (s->rtc.sec == s->rtc.alm_sec) | |
| 122 | + s->status |= 1 << 9; /* RTCALM */ | |
| 123 | + /* TODO: wake-up */ | |
| 124 | + } | |
| 125 | + if (s->rtc.next_comp >= s->rtc.sec) { | |
| 126 | + s->rtc.next -= muldiv64((int16_t) s->rtc.comp, 1000, 0x8000); | |
| 127 | + s->rtc.next_comp = s->rtc.sec + 3600; | |
| 128 | + } | |
| 129 | + menelaus_update(s); | |
| 130 | +} | |
| 131 | + | |
| 132 | +void menelaus_reset(i2c_slave *i2c) | |
| 133 | +{ | |
| 134 | + struct menelaus_s *s = (struct menelaus_s *) i2c; | |
| 135 | + time_t ti; | |
| 136 | + s->reg = 0x00; | |
| 137 | + | |
| 138 | + s->vcore[0] = 0x0c; /* XXX: X-loader needs 0x8c? check! */ | |
| 139 | + s->vcore[1] = 0x05; | |
| 140 | + s->vcore[2] = 0x02; | |
| 141 | + s->vcore[3] = 0x0c; | |
| 142 | + s->vcore[4] = 0x03; | |
| 143 | + s->dcdc[0] = 0x33; /* Depends on wiring */ | |
| 144 | + s->dcdc[1] = 0x03; | |
| 145 | + s->dcdc[2] = 0x00; | |
| 146 | + s->ldo[0] = 0x95; | |
| 147 | + s->ldo[1] = 0x7e; | |
| 148 | + s->ldo[2] = 0x00; | |
| 149 | + s->ldo[3] = 0x00; /* Depends on wiring */ | |
| 150 | + s->ldo[4] = 0x03; /* Depends on wiring */ | |
| 151 | + s->ldo[5] = 0x00; | |
| 152 | + s->ldo[6] = 0x00; | |
| 153 | + s->ldo[7] = 0x00; | |
| 154 | + s->sleep[0] = 0x00; | |
| 155 | + s->sleep[1] = 0x00; | |
| 156 | + s->osc = 0x01; | |
| 157 | + s->detect = 0x09; | |
| 158 | + s->mask = 0x0fff; | |
| 159 | + s->status = 0; | |
| 160 | + s->dir = 0x07; | |
| 161 | + s->outputs = 0x00; | |
| 162 | + s->bbsms = 0x00; | |
| 163 | + s->pull[0] = 0x00; | |
| 164 | + s->pull[1] = 0x00; | |
| 165 | + s->pull[2] = 0x00; | |
| 166 | + s->pull[3] = 0x00; | |
| 167 | + s->mmc_ctrl[0] = 0x03; | |
| 168 | + s->mmc_ctrl[1] = 0xc0; | |
| 169 | + s->mmc_ctrl[2] = 0x00; | |
| 170 | + s->mmc_debounce = 0x05; | |
| 171 | + | |
| 172 | + time(&ti); | |
| 173 | + if (s->rtc.ctrl & 1) | |
| 174 | + menelaus_rtc_stop(s); | |
| 175 | + s->rtc.ctrl = 0x00; | |
| 176 | + s->rtc.comp = 0x0000; | |
| 177 | + s->rtc.next = 1000; | |
| 178 | + s->rtc.sec = ti; | |
| 179 | + s->rtc.next_comp = s->rtc.sec + 1800; | |
| 180 | + s->rtc.alm.tm_sec = 0x00; | |
| 181 | + s->rtc.alm.tm_min = 0x00; | |
| 182 | + s->rtc.alm.tm_hour = 0x00; | |
| 183 | + s->rtc.alm.tm_mday = 0x01; | |
| 184 | + s->rtc.alm.tm_mon = 0x00; | |
| 185 | + s->rtc.alm.tm_year = 2004; | |
| 186 | + menelaus_update(s); | |
| 187 | +} | |
| 188 | + | |
| 189 | +static inline uint8_t to_bcd(int val) | |
| 190 | +{ | |
| 191 | + return ((val / 10) << 4) | (val % 10); | |
| 192 | +} | |
| 193 | + | |
| 194 | +static inline int from_bcd(uint8_t val) | |
| 195 | +{ | |
| 196 | + return ((val >> 4) * 10) + (val & 0x0f); | |
| 197 | +} | |
| 198 | + | |
| 199 | +static void menelaus_gpio_set(void *opaque, int line, int level) | |
| 200 | +{ | |
| 201 | + struct menelaus_s *s = (struct menelaus_s *) opaque; | |
| 202 | + | |
| 203 | + /* No interrupt generated */ | |
| 204 | + s->inputs &= ~(1 << line); | |
| 205 | + s->inputs |= level << line; | |
| 206 | +} | |
| 207 | + | |
| 208 | +static void menelaus_pwrbtn_set(void *opaque, int line, int level) | |
| 209 | +{ | |
| 210 | + struct menelaus_s *s = (struct menelaus_s *) opaque; | |
| 211 | + | |
| 212 | + if (!s->pwrbtn_state && level) { | |
| 213 | + s->status |= 1 << 11; /* PSHBTN */ | |
| 214 | + menelaus_update(s); | |
| 215 | + } | |
| 216 | + s->pwrbtn_state = level; | |
| 217 | +} | |
| 218 | + | |
| 219 | +#define MENELAUS_REV 0x01 | |
| 220 | +#define MENELAUS_VCORE_CTRL1 0x02 | |
| 221 | +#define MENELAUS_VCORE_CTRL2 0x03 | |
| 222 | +#define MENELAUS_VCORE_CTRL3 0x04 | |
| 223 | +#define MENELAUS_VCORE_CTRL4 0x05 | |
| 224 | +#define MENELAUS_VCORE_CTRL5 0x06 | |
| 225 | +#define MENELAUS_DCDC_CTRL1 0x07 | |
| 226 | +#define MENELAUS_DCDC_CTRL2 0x08 | |
| 227 | +#define MENELAUS_DCDC_CTRL3 0x09 | |
| 228 | +#define MENELAUS_LDO_CTRL1 0x0a | |
| 229 | +#define MENELAUS_LDO_CTRL2 0x0b | |
| 230 | +#define MENELAUS_LDO_CTRL3 0x0c | |
| 231 | +#define MENELAUS_LDO_CTRL4 0x0d | |
| 232 | +#define MENELAUS_LDO_CTRL5 0x0e | |
| 233 | +#define MENELAUS_LDO_CTRL6 0x0f | |
| 234 | +#define MENELAUS_LDO_CTRL7 0x10 | |
| 235 | +#define MENELAUS_LDO_CTRL8 0x11 | |
| 236 | +#define MENELAUS_SLEEP_CTRL1 0x12 | |
| 237 | +#define MENELAUS_SLEEP_CTRL2 0x13 | |
| 238 | +#define MENELAUS_DEVICE_OFF 0x14 | |
| 239 | +#define MENELAUS_OSC_CTRL 0x15 | |
| 240 | +#define MENELAUS_DETECT_CTRL 0x16 | |
| 241 | +#define MENELAUS_INT_MASK1 0x17 | |
| 242 | +#define MENELAUS_INT_MASK2 0x18 | |
| 243 | +#define MENELAUS_INT_STATUS1 0x19 | |
| 244 | +#define MENELAUS_INT_STATUS2 0x1a | |
| 245 | +#define MENELAUS_INT_ACK1 0x1b | |
| 246 | +#define MENELAUS_INT_ACK2 0x1c | |
| 247 | +#define MENELAUS_GPIO_CTRL 0x1d | |
| 248 | +#define MENELAUS_GPIO_IN 0x1e | |
| 249 | +#define MENELAUS_GPIO_OUT 0x1f | |
| 250 | +#define MENELAUS_BBSMS 0x20 | |
| 251 | +#define MENELAUS_RTC_CTRL 0x21 | |
| 252 | +#define MENELAUS_RTC_UPDATE 0x22 | |
| 253 | +#define MENELAUS_RTC_SEC 0x23 | |
| 254 | +#define MENELAUS_RTC_MIN 0x24 | |
| 255 | +#define MENELAUS_RTC_HR 0x25 | |
| 256 | +#define MENELAUS_RTC_DAY 0x26 | |
| 257 | +#define MENELAUS_RTC_MON 0x27 | |
| 258 | +#define MENELAUS_RTC_YR 0x28 | |
| 259 | +#define MENELAUS_RTC_WKDAY 0x29 | |
| 260 | +#define MENELAUS_RTC_AL_SEC 0x2a | |
| 261 | +#define MENELAUS_RTC_AL_MIN 0x2b | |
| 262 | +#define MENELAUS_RTC_AL_HR 0x2c | |
| 263 | +#define MENELAUS_RTC_AL_DAY 0x2d | |
| 264 | +#define MENELAUS_RTC_AL_MON 0x2e | |
| 265 | +#define MENELAUS_RTC_AL_YR 0x2f | |
| 266 | +#define MENELAUS_RTC_COMP_MSB 0x30 | |
| 267 | +#define MENELAUS_RTC_COMP_LSB 0x31 | |
| 268 | +#define MENELAUS_S1_PULL_EN 0x32 | |
| 269 | +#define MENELAUS_S1_PULL_DIR 0x33 | |
| 270 | +#define MENELAUS_S2_PULL_EN 0x34 | |
| 271 | +#define MENELAUS_S2_PULL_DIR 0x35 | |
| 272 | +#define MENELAUS_MCT_CTRL1 0x36 | |
| 273 | +#define MENELAUS_MCT_CTRL2 0x37 | |
| 274 | +#define MENELAUS_MCT_CTRL3 0x38 | |
| 275 | +#define MENELAUS_MCT_PIN_ST 0x39 | |
| 276 | +#define MENELAUS_DEBOUNCE1 0x3a | |
| 277 | + | |
| 278 | +static uint8_t menelaus_read(void *opaque, uint8_t addr) | |
| 279 | +{ | |
| 280 | + struct menelaus_s *s = (struct menelaus_s *) opaque; | |
| 281 | + int reg = 0; | |
| 282 | + | |
| 283 | + switch (addr) { | |
| 284 | + case MENELAUS_REV: | |
| 285 | + return 0x22; | |
| 286 | + | |
| 287 | + case MENELAUS_VCORE_CTRL5: reg ++; | |
| 288 | + case MENELAUS_VCORE_CTRL4: reg ++; | |
| 289 | + case MENELAUS_VCORE_CTRL3: reg ++; | |
| 290 | + case MENELAUS_VCORE_CTRL2: reg ++; | |
| 291 | + case MENELAUS_VCORE_CTRL1: | |
| 292 | + return s->vcore[reg]; | |
| 293 | + | |
| 294 | + case MENELAUS_DCDC_CTRL3: reg ++; | |
| 295 | + case MENELAUS_DCDC_CTRL2: reg ++; | |
| 296 | + case MENELAUS_DCDC_CTRL1: | |
| 297 | + return s->dcdc[reg]; | |
| 298 | + | |
| 299 | + case MENELAUS_LDO_CTRL8: reg ++; | |
| 300 | + case MENELAUS_LDO_CTRL7: reg ++; | |
| 301 | + case MENELAUS_LDO_CTRL6: reg ++; | |
| 302 | + case MENELAUS_LDO_CTRL5: reg ++; | |
| 303 | + case MENELAUS_LDO_CTRL4: reg ++; | |
| 304 | + case MENELAUS_LDO_CTRL3: reg ++; | |
| 305 | + case MENELAUS_LDO_CTRL2: reg ++; | |
| 306 | + case MENELAUS_LDO_CTRL1: | |
| 307 | + return s->ldo[reg]; | |
| 308 | + | |
| 309 | + case MENELAUS_SLEEP_CTRL2: reg ++; | |
| 310 | + case MENELAUS_SLEEP_CTRL1: | |
| 311 | + return s->sleep[reg]; | |
| 312 | + | |
| 313 | + case MENELAUS_DEVICE_OFF: | |
| 314 | + return 0; | |
| 315 | + | |
| 316 | + case MENELAUS_OSC_CTRL: | |
| 317 | + return s->osc | (1 << 7); /* CLK32K_GOOD */ | |
| 318 | + | |
| 319 | + case MENELAUS_DETECT_CTRL: | |
| 320 | + return s->detect; | |
| 321 | + | |
| 322 | + case MENELAUS_INT_MASK1: | |
| 323 | + return (s->mask >> 0) & 0xff; | |
| 324 | + case MENELAUS_INT_MASK2: | |
| 325 | + return (s->mask >> 8) & 0xff; | |
| 326 | + | |
| 327 | + case MENELAUS_INT_STATUS1: | |
| 328 | + return (s->status >> 0) & 0xff; | |
| 329 | + case MENELAUS_INT_STATUS2: | |
| 330 | + return (s->status >> 8) & 0xff; | |
| 331 | + | |
| 332 | + case MENELAUS_INT_ACK1: | |
| 333 | + case MENELAUS_INT_ACK2: | |
| 334 | + return 0; | |
| 335 | + | |
| 336 | + case MENELAUS_GPIO_CTRL: | |
| 337 | + return s->dir; | |
| 338 | + case MENELAUS_GPIO_IN: | |
| 339 | + return s->inputs | (~s->dir & s->outputs); | |
| 340 | + case MENELAUS_GPIO_OUT: | |
| 341 | + return s->outputs; | |
| 342 | + | |
| 343 | + case MENELAUS_BBSMS: | |
| 344 | + return s->bbsms; | |
| 345 | + | |
| 346 | + case MENELAUS_RTC_CTRL: | |
| 347 | + return s->rtc.ctrl; | |
| 348 | + case MENELAUS_RTC_UPDATE: | |
| 349 | + return 0x00; | |
| 350 | + case MENELAUS_RTC_SEC: | |
| 351 | + menelaus_rtc_update(s); | |
| 352 | + return to_bcd(s->rtc.tm.tm_sec); | |
| 353 | + case MENELAUS_RTC_MIN: | |
| 354 | + menelaus_rtc_update(s); | |
| 355 | + return to_bcd(s->rtc.tm.tm_min); | |
| 356 | + case MENELAUS_RTC_HR: | |
| 357 | + menelaus_rtc_update(s); | |
| 358 | + if ((s->rtc.ctrl >> 2) & 1) /* MODE12_n24 */ | |
| 359 | + return to_bcd((s->rtc.tm.tm_hour % 12) + 1) | | |
| 360 | + (!!(s->rtc.tm.tm_hour >= 12) << 7); /* PM_nAM */ | |
| 361 | + else | |
| 362 | + return to_bcd(s->rtc.tm.tm_hour); | |
| 363 | + case MENELAUS_RTC_DAY: | |
| 364 | + menelaus_rtc_update(s); | |
| 365 | + return to_bcd(s->rtc.tm.tm_mday); | |
| 366 | + case MENELAUS_RTC_MON: | |
| 367 | + menelaus_rtc_update(s); | |
| 368 | + return to_bcd(s->rtc.tm.tm_mon + 1); | |
| 369 | + case MENELAUS_RTC_YR: | |
| 370 | + menelaus_rtc_update(s); | |
| 371 | + return to_bcd(s->rtc.tm.tm_year - 2000); | |
| 372 | + case MENELAUS_RTC_WKDAY: | |
| 373 | + menelaus_rtc_update(s); | |
| 374 | + return to_bcd(s->rtc.tm.tm_wday); | |
| 375 | + case MENELAUS_RTC_AL_SEC: | |
| 376 | + return to_bcd(s->rtc.alm.tm_sec); | |
| 377 | + case MENELAUS_RTC_AL_MIN: | |
| 378 | + return to_bcd(s->rtc.alm.tm_min); | |
| 379 | + case MENELAUS_RTC_AL_HR: | |
| 380 | + if ((s->rtc.ctrl >> 2) & 1) /* MODE12_n24 */ | |
| 381 | + return to_bcd((s->rtc.alm.tm_hour % 12) + 1) | | |
| 382 | + (!!(s->rtc.alm.tm_hour >= 12) << 7);/* AL_PM_nAM */ | |
| 383 | + else | |
| 384 | + return to_bcd(s->rtc.alm.tm_hour); | |
| 385 | + case MENELAUS_RTC_AL_DAY: | |
| 386 | + return to_bcd(s->rtc.alm.tm_mday); | |
| 387 | + case MENELAUS_RTC_AL_MON: | |
| 388 | + return to_bcd(s->rtc.alm.tm_mon + 1); | |
| 389 | + case MENELAUS_RTC_AL_YR: | |
| 390 | + return to_bcd(s->rtc.alm.tm_year - 2000); | |
| 391 | + case MENELAUS_RTC_COMP_MSB: | |
| 392 | + return (s->rtc.comp >> 8) & 0xff; | |
| 393 | + case MENELAUS_RTC_COMP_LSB: | |
| 394 | + return (s->rtc.comp >> 0) & 0xff; | |
| 395 | + | |
| 396 | + case MENELAUS_S1_PULL_EN: | |
| 397 | + return s->pull[0]; | |
| 398 | + case MENELAUS_S1_PULL_DIR: | |
| 399 | + return s->pull[1]; | |
| 400 | + case MENELAUS_S2_PULL_EN: | |
| 401 | + return s->pull[2]; | |
| 402 | + case MENELAUS_S2_PULL_DIR: | |
| 403 | + return s->pull[3]; | |
| 404 | + | |
| 405 | + case MENELAUS_MCT_CTRL3: reg ++; | |
| 406 | + case MENELAUS_MCT_CTRL2: reg ++; | |
| 407 | + case MENELAUS_MCT_CTRL1: | |
| 408 | + return s->mmc_ctrl[reg]; | |
| 409 | + case MENELAUS_MCT_PIN_ST: | |
| 410 | + /* TODO: return the real Card Detect */ | |
| 411 | + return 0; | |
| 412 | + case MENELAUS_DEBOUNCE1: | |
| 413 | + return s->mmc_debounce; | |
| 414 | + | |
| 415 | + default: | |
| 416 | +#ifdef VERBOSE | |
| 417 | + printf("%s: unknown register %02x\n", __FUNCTION__, addr); | |
| 418 | +#endif | |
| 419 | + break; | |
| 420 | + } | |
| 421 | + return 0; | |
| 422 | +} | |
| 423 | + | |
| 424 | +static void menelaus_write(void *opaque, uint8_t addr, uint8_t value) | |
| 425 | +{ | |
| 426 | + struct menelaus_s *s = (struct menelaus_s *) opaque; | |
| 427 | + int line; | |
| 428 | + int reg = 0; | |
| 429 | + struct tm tm; | |
| 430 | + | |
| 431 | + switch (addr) { | |
| 432 | + case MENELAUS_VCORE_CTRL1: | |
| 433 | + s->vcore[0] = (value & 0xe) | MIN(value & 0x1f, 0x12); | |
| 434 | + break; | |
| 435 | + case MENELAUS_VCORE_CTRL2: | |
| 436 | + s->vcore[1] = value; | |
| 437 | + break; | |
| 438 | + case MENELAUS_VCORE_CTRL3: | |
| 439 | + s->vcore[2] = MIN(value & 0x1f, 0x12); | |
| 440 | + break; | |
| 441 | + case MENELAUS_VCORE_CTRL4: | |
| 442 | + s->vcore[3] = MIN(value & 0x1f, 0x12); | |
| 443 | + break; | |
| 444 | + case MENELAUS_VCORE_CTRL5: | |
| 445 | + s->vcore[4] = value & 3; | |
| 446 | + /* XXX | |
| 447 | + * auto set to 3 on M_Active, nRESWARM | |
| 448 | + * auto set to 0 on M_WaitOn, M_Backup | |
| 449 | + */ | |
| 450 | + break; | |
| 451 | + | |
| 452 | + case MENELAUS_DCDC_CTRL1: | |
| 453 | + s->dcdc[0] = value & 0x3f; | |
| 454 | + break; | |
| 455 | + case MENELAUS_DCDC_CTRL2: | |
| 456 | + s->dcdc[1] = value & 0x07; | |
| 457 | + /* XXX | |
| 458 | + * auto set to 3 on M_Active, nRESWARM | |
| 459 | + * auto set to 0 on M_WaitOn, M_Backup | |
| 460 | + */ | |
| 461 | + break; | |
| 462 | + case MENELAUS_DCDC_CTRL3: | |
| 463 | + s->dcdc[2] = value & 0x07; | |
| 464 | + break; | |
| 465 | + | |
| 466 | + case MENELAUS_LDO_CTRL1: | |
| 467 | + s->ldo[0] = value; | |
| 468 | + break; | |
| 469 | + case MENELAUS_LDO_CTRL2: | |
| 470 | + s->ldo[1] = value & 0x7f; | |
| 471 | + /* XXX | |
| 472 | + * auto set to 0x7e on M_WaitOn, M_Backup | |
| 473 | + */ | |
| 474 | + break; | |
| 475 | + case MENELAUS_LDO_CTRL3: | |
| 476 | + s->ldo[2] = value & 3; | |
| 477 | + /* XXX | |
| 478 | + * auto set to 3 on M_Active, nRESWARM | |
| 479 | + * auto set to 0 on M_WaitOn, M_Backup | |
| 480 | + */ | |
| 481 | + break; | |
| 482 | + case MENELAUS_LDO_CTRL4: | |
| 483 | + s->ldo[3] = value & 3; | |
| 484 | + /* XXX | |
| 485 | + * auto set to 3 on M_Active, nRESWARM | |
| 486 | + * auto set to 0 on M_WaitOn, M_Backup | |
| 487 | + */ | |
| 488 | + break; | |
| 489 | + case MENELAUS_LDO_CTRL5: | |
| 490 | + s->ldo[4] = value & 3; | |
| 491 | + /* XXX | |
| 492 | + * auto set to 3 on M_Active, nRESWARM | |
| 493 | + * auto set to 0 on M_WaitOn, M_Backup | |
| 494 | + */ | |
| 495 | + break; | |
| 496 | + case MENELAUS_LDO_CTRL6: | |
| 497 | + s->ldo[5] = value & 3; | |
| 498 | + break; | |
| 499 | + case MENELAUS_LDO_CTRL7: | |
| 500 | + s->ldo[6] = value & 3; | |
| 501 | + break; | |
| 502 | + case MENELAUS_LDO_CTRL8: | |
| 503 | + s->ldo[7] = value & 3; | |
| 504 | + break; | |
| 505 | + | |
| 506 | + case MENELAUS_SLEEP_CTRL2: reg ++; | |
| 507 | + case MENELAUS_SLEEP_CTRL1: | |
| 508 | + s->sleep[reg] = value; | |
| 509 | + break; | |
| 510 | + | |
| 511 | + case MENELAUS_DEVICE_OFF: | |
| 512 | + if (value & 1) | |
| 513 | + menelaus_reset(&s->i2c); | |
| 514 | + break; | |
| 515 | + | |
| 516 | + case MENELAUS_OSC_CTRL: | |
| 517 | + s->osc = value & 7; | |
| 518 | + break; | |
| 519 | + | |
| 520 | + case MENELAUS_DETECT_CTRL: | |
| 521 | + s->detect = value & 0x7f; | |
| 522 | + break; | |
| 523 | + | |
| 524 | + case MENELAUS_INT_MASK1: | |
| 525 | + s->mask &= 0xf00; | |
| 526 | + s->mask |= value << 0; | |
| 527 | + menelaus_update(s); | |
| 528 | + break; | |
| 529 | + case MENELAUS_INT_MASK2: | |
| 530 | + s->mask &= 0x0ff; | |
| 531 | + s->mask |= value << 8; | |
| 532 | + menelaus_update(s); | |
| 533 | + break; | |
| 534 | + | |
| 535 | + case MENELAUS_INT_ACK1: | |
| 536 | + s->status &= ~(((uint16_t) value) << 0); | |
| 537 | + menelaus_update(s); | |
| 538 | + break; | |
| 539 | + case MENELAUS_INT_ACK2: | |
| 540 | + s->status &= ~(((uint16_t) value) << 8); | |
| 541 | + menelaus_update(s); | |
| 542 | + break; | |
| 543 | + | |
| 544 | + case MENELAUS_GPIO_CTRL: | |
| 545 | + for (line = 0; line < 3; line ++) | |
| 546 | + if (((s->dir ^ value) >> line) & 1) | |
| 547 | + if (s->handler[line]) | |
| 548 | + qemu_set_irq(s->handler[line], | |
| 549 | + ((s->outputs & ~s->dir) >> line) & 1); | |
| 550 | + s->dir = value & 0x67; | |
| 551 | + break; | |
| 552 | + case MENELAUS_GPIO_OUT: | |
| 553 | + for (line = 0; line < 3; line ++) | |
| 554 | + if ((((s->outputs ^ value) & ~s->dir) >> line) & 1) | |
| 555 | + if (s->handler[line]) | |
| 556 | + qemu_set_irq(s->handler[line], (s->outputs >> line) & 1); | |
| 557 | + s->outputs = value & 0x07; | |
| 558 | + break; | |
| 559 | + | |
| 560 | + case MENELAUS_BBSMS: | |
| 561 | + s->bbsms = 0x0d; | |
| 562 | + break; | |
| 563 | + | |
| 564 | + case MENELAUS_RTC_CTRL: | |
| 565 | + if ((s->rtc.ctrl ^ value) & 1) { /* RTC_EN */ | |
| 566 | + if (value & 1) | |
| 567 | + menelaus_rtc_start(s); | |
| 568 | + else | |
| 569 | + menelaus_rtc_stop(s); | |
| 570 | + } | |
| 571 | + s->rtc.ctrl = value & 0x1f; | |
| 572 | + menelaus_alm_update(s); | |
| 573 | + break; | |
| 574 | + case MENELAUS_RTC_UPDATE: | |
| 575 | + menelaus_rtc_update(s); | |
| 576 | + memcpy(&tm, &s->rtc.tm, sizeof(tm)); | |
| 577 | + switch (value & 0xf) { | |
| 578 | + case 0: | |
| 579 | + break; | |
| 580 | + case 1: | |
| 581 | + tm.tm_sec = s->rtc.new.tm_sec; | |
| 582 | + break; | |
| 583 | + case 2: | |
| 584 | + tm.tm_min = s->rtc.new.tm_min; | |
| 585 | + break; | |
| 586 | + case 3: | |
| 587 | + if (s->rtc.new.tm_hour > 23) | |
| 588 | + goto rtc_badness; | |
| 589 | + tm.tm_hour = s->rtc.new.tm_hour; | |
| 590 | + break; | |
| 591 | + case 4: | |
| 592 | + if (s->rtc.new.tm_mday < 1) | |
| 593 | + goto rtc_badness; | |
| 594 | + /* TODO check range */ | |
| 595 | + tm.tm_mday = s->rtc.new.tm_mday; | |
| 596 | + break; | |
| 597 | + case 5: | |
| 598 | + if (s->rtc.new.tm_mon < 0 || s->rtc.new.tm_mon > 11) | |
| 599 | + goto rtc_badness; | |
| 600 | + tm.tm_mon = s->rtc.new.tm_mon; | |
| 601 | + break; | |
| 602 | + case 6: | |
| 603 | + tm.tm_year = s->rtc.new.tm_year; | |
| 604 | + break; | |
| 605 | + case 7: | |
| 606 | + /* TODO set .tm_mday instead */ | |
| 607 | + tm.tm_wday = s->rtc.new.tm_wday; | |
| 608 | + break; | |
| 609 | + case 8: | |
| 610 | + if (s->rtc.new.tm_hour > 23) | |
| 611 | + goto rtc_badness; | |
| 612 | + if (s->rtc.new.tm_mday < 1) | |
| 613 | + goto rtc_badness; | |
| 614 | + if (s->rtc.new.tm_mon < 0 || s->rtc.new.tm_mon > 11) | |
| 615 | + goto rtc_badness; | |
| 616 | + tm.tm_sec = s->rtc.new.tm_sec; | |
| 617 | + tm.tm_min = s->rtc.new.tm_min; | |
| 618 | + tm.tm_hour = s->rtc.new.tm_hour; | |
| 619 | + tm.tm_mday = s->rtc.new.tm_mday; | |
| 620 | + tm.tm_mon = s->rtc.new.tm_mon; | |
| 621 | + tm.tm_year = s->rtc.new.tm_year; | |
| 622 | + break; | |
| 623 | + rtc_badness: | |
| 624 | + default: | |
| 625 | + fprintf(stderr, "%s: bad RTC_UPDATE value %02x\n", | |
| 626 | + __FUNCTION__, value); | |
| 627 | + s->status |= 1 << 10; /* RTCERR */ | |
| 628 | + menelaus_update(s); | |
| 629 | + } | |
| 630 | + s->rtc.sec += difftime(mktime(&tm), mktime(&s->rtc.tm)); | |
| 631 | + break; | |
| 632 | + case MENELAUS_RTC_SEC: | |
| 633 | + s->rtc.tm.tm_sec = from_bcd(value & 0x7f); | |
| 634 | + break; | |
| 635 | + case MENELAUS_RTC_MIN: | |
| 636 | + s->rtc.tm.tm_min = from_bcd(value & 0x7f); | |
| 637 | + break; | |
| 638 | + case MENELAUS_RTC_HR: | |
| 639 | + s->rtc.tm.tm_hour = (s->rtc.ctrl & (1 << 2)) ? /* MODE12_n24 */ | |
| 640 | + MIN(from_bcd(value & 0x3f), 12) + ((value >> 7) ? 11 : -1) : | |
| 641 | + from_bcd(value & 0x3f); | |
| 642 | + break; | |
| 643 | + case MENELAUS_RTC_DAY: | |
| 644 | + s->rtc.tm.tm_mday = from_bcd(value); | |
| 645 | + break; | |
| 646 | + case MENELAUS_RTC_MON: | |
| 647 | + s->rtc.tm.tm_mon = MAX(1, from_bcd(value)) - 1; | |
| 648 | + break; | |
| 649 | + case MENELAUS_RTC_YR: | |
| 650 | + s->rtc.tm.tm_year = 2000 + from_bcd(value); | |
| 651 | + break; | |
| 652 | + case MENELAUS_RTC_WKDAY: | |
| 653 | + s->rtc.tm.tm_mday = from_bcd(value); | |
| 654 | + break; | |
| 655 | + case MENELAUS_RTC_AL_SEC: | |
| 656 | + s->rtc.alm.tm_sec = from_bcd(value & 0x7f); | |
| 657 | + menelaus_alm_update(s); | |
| 658 | + break; | |
| 659 | + case MENELAUS_RTC_AL_MIN: | |
| 660 | + s->rtc.alm.tm_min = from_bcd(value & 0x7f); | |
| 661 | + menelaus_alm_update(s); | |
| 662 | + break; | |
| 663 | + case MENELAUS_RTC_AL_HR: | |
| 664 | + s->rtc.alm.tm_hour = (s->rtc.ctrl & (1 << 2)) ? /* MODE12_n24 */ | |
| 665 | + MIN(from_bcd(value & 0x3f), 12) + ((value >> 7) ? 11 : -1) : | |
| 666 | + from_bcd(value & 0x3f); | |
| 667 | + menelaus_alm_update(s); | |
| 668 | + break; | |
| 669 | + case MENELAUS_RTC_AL_DAY: | |
| 670 | + s->rtc.alm.tm_mday = from_bcd(value); | |
| 671 | + menelaus_alm_update(s); | |
| 672 | + break; | |
| 673 | + case MENELAUS_RTC_AL_MON: | |
| 674 | + s->rtc.alm.tm_mon = MAX(1, from_bcd(value)) - 1; | |
| 675 | + menelaus_alm_update(s); | |
| 676 | + break; | |
| 677 | + case MENELAUS_RTC_AL_YR: | |
| 678 | + s->rtc.alm.tm_year = 2000 + from_bcd(value); | |
| 679 | + menelaus_alm_update(s); | |
| 680 | + break; | |
| 681 | + case MENELAUS_RTC_COMP_MSB: | |
| 682 | + s->rtc.comp &= 0xff; | |
| 683 | + s->rtc.comp |= value << 8; | |
| 684 | + break; | |
| 685 | + case MENELAUS_RTC_COMP_LSB: | |
| 686 | + s->rtc.comp &= 0xff << 8; | |
| 687 | + s->rtc.comp |= value; | |
| 688 | + break; | |
| 689 | + | |
| 690 | + case MENELAUS_S1_PULL_EN: | |
| 691 | + s->pull[0] = value; | |
| 692 | + break; | |
| 693 | + case MENELAUS_S1_PULL_DIR: | |
| 694 | + s->pull[1] = value & 0x1f; | |
| 695 | + break; | |
| 696 | + case MENELAUS_S2_PULL_EN: | |
| 697 | + s->pull[2] = value; | |
| 698 | + break; | |
| 699 | + case MENELAUS_S2_PULL_DIR: | |
| 700 | + s->pull[3] = value & 0x1f; | |
| 701 | + break; | |
| 702 | + | |
| 703 | + case MENELAUS_MCT_CTRL1: | |
| 704 | + s->mmc_ctrl[0] = value & 0x7f; | |
| 705 | + break; | |
| 706 | + case MENELAUS_MCT_CTRL2: | |
| 707 | + s->mmc_ctrl[1] = value; | |
| 708 | + /* TODO update Card Detect interrupts */ | |
| 709 | + break; | |
| 710 | + case MENELAUS_MCT_CTRL3: | |
| 711 | + s->mmc_ctrl[2] = value & 0xf; | |
| 712 | + break; | |
| 713 | + case MENELAUS_DEBOUNCE1: | |
| 714 | + s->mmc_debounce = value & 0x3f; | |
| 715 | + break; | |
| 716 | + | |
| 717 | + default: | |
| 718 | +#ifdef VERBOSE | |
| 719 | + printf("%s: unknown register %02x\n", __FUNCTION__, addr); | |
| 720 | +#endif | |
| 721 | + } | |
| 722 | +} | |
| 723 | + | |
| 724 | +static void menelaus_event(i2c_slave *i2c, enum i2c_event event) | |
| 725 | +{ | |
| 726 | + struct menelaus_s *s = (struct menelaus_s *) i2c; | |
| 727 | + | |
| 728 | + if (event == I2C_START_SEND) | |
| 729 | + s->firstbyte = 1; | |
| 730 | +} | |
| 731 | + | |
| 732 | +static int menelaus_tx(i2c_slave *i2c, uint8_t data) | |
| 733 | +{ | |
| 734 | + struct menelaus_s *s = (struct menelaus_s *) i2c; | |
| 735 | + /* Interpret register address byte */ | |
| 736 | + if (s->firstbyte) { | |
| 737 | + s->reg = data; | |
| 738 | + s->firstbyte = 0; | |
| 739 | + } else | |
| 740 | + menelaus_write(s, s->reg ++, data); | |
| 741 | + | |
| 742 | + return 0; | |
| 743 | +} | |
| 744 | + | |
| 745 | +static int menelaus_rx(i2c_slave *i2c) | |
| 746 | +{ | |
| 747 | + struct menelaus_s *s = (struct menelaus_s *) i2c; | |
| 748 | + | |
| 749 | + return menelaus_read(s, s->reg ++); | |
| 750 | +} | |
| 751 | + | |
| 752 | +static void tm_put(QEMUFile *f, struct tm *tm) { | |
| 753 | + qemu_put_be16(f, tm->tm_sec); | |
| 754 | + qemu_put_be16(f, tm->tm_min); | |
| 755 | + qemu_put_be16(f, tm->tm_hour); | |
| 756 | + qemu_put_be16(f, tm->tm_mday); | |
| 757 | + qemu_put_be16(f, tm->tm_min); | |
| 758 | + qemu_put_be16(f, tm->tm_year); | |
| 759 | +} | |
| 760 | + | |
| 761 | +static void tm_get(QEMUFile *f, struct tm *tm) { | |
| 762 | + tm->tm_sec = qemu_get_be16(f); | |
| 763 | + tm->tm_min = qemu_get_be16(f); | |
| 764 | + tm->tm_hour = qemu_get_be16(f); | |
| 765 | + tm->tm_mday = qemu_get_be16(f); | |
| 766 | + tm->tm_min = qemu_get_be16(f); | |
| 767 | + tm->tm_year = qemu_get_be16(f); | |
| 768 | +} | |
| 769 | + | |
| 770 | +static void menelaus_save(QEMUFile *f, void *opaque) | |
| 771 | +{ | |
| 772 | + struct menelaus_s *s = (struct menelaus_s *) opaque; | |
| 773 | + | |
| 774 | + qemu_put_be32(f, s->firstbyte); | |
| 775 | + qemu_put_8s(f, &s->reg); | |
| 776 | + | |
| 777 | + qemu_put_8s(f, &s->vcore[0]); | |
| 778 | + qemu_put_8s(f, &s->vcore[1]); | |
| 779 | + qemu_put_8s(f, &s->vcore[2]); | |
| 780 | + qemu_put_8s(f, &s->vcore[3]); | |
| 781 | + qemu_put_8s(f, &s->vcore[4]); | |
| 782 | + qemu_put_8s(f, &s->dcdc[3]); | |
| 783 | + qemu_put_8s(f, &s->dcdc[3]); | |
| 784 | + qemu_put_8s(f, &s->dcdc[3]); | |
| 785 | + qemu_put_8s(f, &s->ldo[0]); | |
| 786 | + qemu_put_8s(f, &s->ldo[1]); | |
| 787 | + qemu_put_8s(f, &s->ldo[2]); | |
| 788 | + qemu_put_8s(f, &s->ldo[3]); | |
| 789 | + qemu_put_8s(f, &s->ldo[4]); | |
| 790 | + qemu_put_8s(f, &s->ldo[5]); | |
| 791 | + qemu_put_8s(f, &s->ldo[6]); | |
| 792 | + qemu_put_8s(f, &s->ldo[7]); | |
| 793 | + qemu_put_8s(f, &s->sleep[0]); | |
| 794 | + qemu_put_8s(f, &s->sleep[1]); | |
| 795 | + qemu_put_8s(f, &s->osc); | |
| 796 | + qemu_put_8s(f, &s->detect); | |
| 797 | + qemu_put_be16s(f, &s->mask); | |
| 798 | + qemu_put_be16s(f, &s->status); | |
| 799 | + qemu_put_8s(f, &s->dir); | |
| 800 | + qemu_put_8s(f, &s->inputs); | |
| 801 | + qemu_put_8s(f, &s->outputs); | |
| 802 | + qemu_put_8s(f, &s->bbsms); | |
| 803 | + qemu_put_8s(f, &s->pull[0]); | |
| 804 | + qemu_put_8s(f, &s->pull[1]); | |
| 805 | + qemu_put_8s(f, &s->pull[2]); | |
| 806 | + qemu_put_8s(f, &s->pull[3]); | |
| 807 | + qemu_put_8s(f, &s->mmc_ctrl[0]); | |
| 808 | + qemu_put_8s(f, &s->mmc_ctrl[1]); | |
| 809 | + qemu_put_8s(f, &s->mmc_ctrl[2]); | |
| 810 | + qemu_put_8s(f, &s->mmc_debounce); | |
| 811 | + qemu_put_8s(f, &s->rtc.ctrl); | |
| 812 | + qemu_put_be16s(f, &s->rtc.comp); | |
| 813 | + /* Should be <= 1000 */ | |
| 814 | + qemu_put_be16(f, s->rtc.next - qemu_get_clock(rt_clock)); | |
| 815 | + tm_put(f, &s->rtc.new); | |
| 816 | + tm_put(f, &s->rtc.alm); | |
| 817 | + qemu_put_byte(f, s->pwrbtn_state); | |
| 818 | + | |
| 819 | + i2c_slave_save(f, &s->i2c); | |
| 820 | +} | |
| 821 | + | |
| 822 | +static int menelaus_load(QEMUFile *f, void *opaque, int version_id) | |
| 823 | +{ | |
| 824 | + struct menelaus_s *s = (struct menelaus_s *) opaque; | |
| 825 | + | |
| 826 | + s->firstbyte = qemu_get_be32(f); | |
| 827 | + qemu_get_8s(f, &s->reg); | |
| 828 | + | |
| 829 | + if (s->rtc.ctrl & 1) /* RTC_EN */ | |
| 830 | + menelaus_rtc_stop(s); | |
| 831 | + qemu_get_8s(f, &s->vcore[0]); | |
| 832 | + qemu_get_8s(f, &s->vcore[1]); | |
| 833 | + qemu_get_8s(f, &s->vcore[2]); | |
| 834 | + qemu_get_8s(f, &s->vcore[3]); | |
| 835 | + qemu_get_8s(f, &s->vcore[4]); | |
| 836 | + qemu_get_8s(f, &s->dcdc[3]); | |
| 837 | + qemu_get_8s(f, &s->dcdc[3]); | |
| 838 | + qemu_get_8s(f, &s->dcdc[3]); | |
| 839 | + qemu_get_8s(f, &s->ldo[0]); | |
| 840 | + qemu_get_8s(f, &s->ldo[1]); | |
| 841 | + qemu_get_8s(f, &s->ldo[2]); | |
| 842 | + qemu_get_8s(f, &s->ldo[3]); | |
| 843 | + qemu_get_8s(f, &s->ldo[4]); | |
| 844 | + qemu_get_8s(f, &s->ldo[5]); | |
| 845 | + qemu_get_8s(f, &s->ldo[6]); | |
| 846 | + qemu_get_8s(f, &s->ldo[7]); | |
| 847 | + qemu_get_8s(f, &s->sleep[0]); | |
| 848 | + qemu_get_8s(f, &s->sleep[1]); | |
| 849 | + qemu_get_8s(f, &s->osc); | |
| 850 | + qemu_get_8s(f, &s->detect); | |
| 851 | + qemu_get_be16s(f, &s->mask); | |
| 852 | + qemu_get_be16s(f, &s->status); | |
| 853 | + qemu_get_8s(f, &s->dir); | |
| 854 | + qemu_get_8s(f, &s->inputs); | |
| 855 | + qemu_get_8s(f, &s->outputs); | |
| 856 | + qemu_get_8s(f, &s->bbsms); | |
| 857 | + qemu_get_8s(f, &s->pull[0]); | |
| 858 | + qemu_get_8s(f, &s->pull[1]); | |
| 859 | + qemu_get_8s(f, &s->pull[2]); | |
| 860 | + qemu_get_8s(f, &s->pull[3]); | |
| 861 | + qemu_get_8s(f, &s->mmc_ctrl[0]); | |
| 862 | + qemu_get_8s(f, &s->mmc_ctrl[1]); | |
| 863 | + qemu_get_8s(f, &s->mmc_ctrl[2]); | |
| 864 | + qemu_get_8s(f, &s->mmc_debounce); | |
| 865 | + qemu_get_8s(f, &s->rtc.ctrl); | |
| 866 | + qemu_get_be16s(f, &s->rtc.comp); | |
| 867 | + s->rtc.next = qemu_get_be16(f); | |
| 868 | + tm_get(f, &s->rtc.new); | |
| 869 | + tm_get(f, &s->rtc.alm); | |
| 870 | + s->pwrbtn_state = qemu_get_byte(f); | |
| 871 | + menelaus_alm_update(s); | |
| 872 | + menelaus_update(s); | |
| 873 | + if (s->rtc.ctrl & 1) /* RTC_EN */ | |
| 874 | + menelaus_rtc_start(s); | |
| 875 | + | |
| 876 | + i2c_slave_load(f, &s->i2c); | |
| 877 | + return 0; | |
| 878 | +} | |
| 879 | + | |
| 880 | +static int menelaus_iid = 0; | |
| 881 | + | |
| 882 | +i2c_slave *twl92230_init(i2c_bus *bus, qemu_irq irq) | |
| 883 | +{ | |
| 884 | + struct menelaus_s *s = (struct menelaus_s *) | |
| 885 | + i2c_slave_init(bus, 0, sizeof(struct menelaus_s)); | |
| 886 | + | |
| 887 | + s->i2c.event = menelaus_event; | |
| 888 | + s->i2c.recv = menelaus_rx; | |
| 889 | + s->i2c.send = menelaus_tx; | |
| 890 | + | |
| 891 | + /* TODO: use the qemu gettime functions */ | |
| 892 | + s->rtc.gettime = localtime_r; | |
| 893 | + | |
| 894 | + s->irq = irq; | |
| 895 | + s->rtc.hz = qemu_new_timer(rt_clock, menelaus_rtc_hz, s); | |
| 896 | + s->in = qemu_allocate_irqs(menelaus_gpio_set, s, 3); | |
| 897 | + s->pwrbtn = qemu_allocate_irqs(menelaus_pwrbtn_set, s, 1)[0]; | |
| 898 | + | |
| 899 | + menelaus_reset(&s->i2c); | |
| 900 | + | |
| 901 | + register_savevm("menelaus", menelaus_iid ++, | |
| 902 | + 0, menelaus_save, menelaus_load, s); | |
| 903 | + | |
| 904 | + return &s->i2c; | |
| 905 | +} | |
| 906 | + | |
| 907 | +qemu_irq *twl92230_gpio_in_get(i2c_slave *i2c) | |
| 908 | +{ | |
| 909 | + struct menelaus_s *s = (struct menelaus_s *) i2c; | |
| 910 | + | |
| 911 | + return s->in; | |
| 912 | +} | |
| 913 | + | |
| 914 | +void twl92230_gpio_out_set(i2c_slave *i2c, int line, qemu_irq handler) | |
| 915 | +{ | |
| 916 | + struct menelaus_s *s = (struct menelaus_s *) i2c; | |
| 917 | + | |
| 918 | + if (line >= 3 || line < 0) { | |
| 919 | + fprintf(stderr, "%s: No GPO line %i\n", __FUNCTION__, line); | |
| 920 | + exit(-1); | |
| 921 | + } | |
| 922 | + s->handler[line] = handler; | |
| 923 | +} | ... | ... |
vl.c
| ... | ... | @@ -8051,6 +8051,7 @@ static void register_machines(void) |
| 8051 | 8051 | qemu_register_machine(&borzoipda_machine); |
| 8052 | 8052 | qemu_register_machine(&terrierpda_machine); |
| 8053 | 8053 | qemu_register_machine(&palmte_machine); |
| 8054 | + qemu_register_machine(&n800_machine); | |
| 8054 | 8055 | qemu_register_machine(&lm3s811evb_machine); |
| 8055 | 8056 | qemu_register_machine(&lm3s6965evb_machine); |
| 8056 | 8057 | qemu_register_machine(&connex_machine); | ... | ... |