Commit c3d2689d88159291ef8af338b5f01cfbe5551d2c
1 parent
a5236105
Basic OMAP310 support. Basic Palm Tungsten|E machine emulation.
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@3091 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
7 changed files
with
4914 additions
and
1 deletions
Too many changes to show.
To preserve performance only 7 of 12 files are displayed.
Makefile.target
| ... | ... | @@ -463,6 +463,7 @@ VL_OBJS+= arm-semi.o |
| 463 | 463 | VL_OBJS+= pxa2xx.o pxa2xx_pic.o pxa2xx_gpio.o pxa2xx_timer.o pxa2xx_dma.o |
| 464 | 464 | VL_OBJS+= pxa2xx_lcd.o pxa2xx_mmci.o pxa2xx_pcmcia.o max111x.o max7310.o |
| 465 | 465 | VL_OBJS+= spitz.o ads7846.o ide.o serial.o nand.o $(AUDIODRV) wm8750.o |
| 466 | +VL_OBJS+= omap.o omap_lcdc.o omap1_clk.o palm.o | |
| 466 | 467 | CPPFLAGS += -DHAS_AUDIO |
| 467 | 468 | endif |
| 468 | 469 | ifeq ($(TARGET_BASE_ARCH), sh4) | ... | ... |
cpu-all.h
| ... | ... | @@ -702,7 +702,8 @@ void cpu_dump_statistics (CPUState *env, FILE *f, |
| 702 | 702 | int flags); |
| 703 | 703 | |
| 704 | 704 | void cpu_abort(CPUState *env, const char *fmt, ...) |
| 705 | - __attribute__ ((__format__ (__printf__, 2, 3))); | |
| 705 | + __attribute__ ((__format__ (__printf__, 2, 3))) | |
| 706 | + __attribute__ ((__noreturn__)); | |
| 706 | 707 | extern CPUState *first_cpu; |
| 707 | 708 | extern CPUState *cpu_single_env; |
| 708 | 709 | extern int code_copy_enabled; | ... | ... |
hw/omap.c
0 → 100644
| 1 | +/* | |
| 2 | + * TI OMAP processors emulation. | |
| 3 | + * | |
| 4 | + * Copyright (C) 2006-2007 Andrzej Zaborowski <balrog@zabor.org> | |
| 5 | + * | |
| 6 | + * This program is free software; you can redistribute it and/or | |
| 7 | + * modify it under the terms of the GNU General Public License as | |
| 8 | + * published by the Free Software Foundation; either version 2 of | |
| 9 | + * the License, or (at your option) any later version. | |
| 10 | + * | |
| 11 | + * This program is distributed in the hope that it will be useful, | |
| 12 | + * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
| 13 | + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
| 14 | + * GNU General Public License for more details. | |
| 15 | + * | |
| 16 | + * You should have received a copy of the GNU General Public License | |
| 17 | + * along with this program; if not, write to the Free Software | |
| 18 | + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, | |
| 19 | + * MA 02111-1307 USA | |
| 20 | + */ | |
| 21 | +#include "vl.h" | |
| 22 | +#include "arm_pic.h" | |
| 23 | + | |
| 24 | +/* Should signal the TCMI */ | |
| 25 | +static uint32_t omap_badwidth_read16(void *opaque, target_phys_addr_t addr) | |
| 26 | +{ | |
| 27 | + OMAP_16B_REG(addr); | |
| 28 | + return 0; | |
| 29 | +} | |
| 30 | + | |
| 31 | +static void omap_badwidth_write16(void *opaque, target_phys_addr_t addr, | |
| 32 | + uint32_t value) | |
| 33 | +{ | |
| 34 | + OMAP_16B_REG(addr); | |
| 35 | +} | |
| 36 | + | |
| 37 | +static uint32_t omap_badwidth_read32(void *opaque, target_phys_addr_t addr) | |
| 38 | +{ | |
| 39 | + OMAP_32B_REG(addr); | |
| 40 | + return 0; | |
| 41 | +} | |
| 42 | + | |
| 43 | +static void omap_badwidth_write32(void *opaque, target_phys_addr_t addr, | |
| 44 | + uint32_t value) | |
| 45 | +{ | |
| 46 | + OMAP_32B_REG(addr); | |
| 47 | +} | |
| 48 | + | |
| 49 | +#define likely | |
| 50 | +#define unlikely | |
| 51 | + | |
| 52 | +/* Interrupt Handlers */ | |
| 53 | +struct omap_intr_handler_s { | |
| 54 | + qemu_irq *pins; | |
| 55 | + qemu_irq *parent_pic; | |
| 56 | + target_phys_addr_t base; | |
| 57 | + | |
| 58 | + /* state */ | |
| 59 | + uint32_t irqs; | |
| 60 | + uint32_t mask; | |
| 61 | + uint32_t sens_edge; | |
| 62 | + uint32_t fiq; | |
| 63 | + int priority[32]; | |
| 64 | + uint32_t new_irq_agr; | |
| 65 | + uint32_t new_fiq_agr; | |
| 66 | + int sir_irq; | |
| 67 | + int sir_fiq; | |
| 68 | + int stats[32]; | |
| 69 | +}; | |
| 70 | + | |
| 71 | +static void omap_inth_update(struct omap_intr_handler_s *s) | |
| 72 | +{ | |
| 73 | + uint32_t irq = s->new_irq_agr & s->irqs & ~s->mask & ~s->fiq; | |
| 74 | + uint32_t fiq = s->new_fiq_agr & s->irqs & ~s->mask & s->fiq; | |
| 75 | + | |
| 76 | + qemu_set_irq(s->parent_pic[ARM_PIC_CPU_IRQ], irq); | |
| 77 | + if (irq) | |
| 78 | + s->new_irq_agr = 0; | |
| 79 | + | |
| 80 | + qemu_set_irq(s->parent_pic[ARM_PIC_CPU_FIQ], fiq); | |
| 81 | + if (fiq) | |
| 82 | + s->new_fiq_agr = 0; | |
| 83 | +} | |
| 84 | + | |
| 85 | +static void omap_inth_sir_update(struct omap_intr_handler_s *s) | |
| 86 | +{ | |
| 87 | + int i, intr_irq, intr_fiq, p_irq, p_fiq, p, f; | |
| 88 | + uint32_t level = s->irqs & ~s->mask; | |
| 89 | + | |
| 90 | + intr_irq = 0; | |
| 91 | + intr_fiq = 0; | |
| 92 | + p_irq = -1; | |
| 93 | + p_fiq = -1; | |
| 94 | + /* Find the interrupt line with the highest dynamic priority */ | |
| 95 | + for (f = ffs(level), i = f - 1, level >>= f - 1; f; i += f, level >>= f) { | |
| 96 | + p = s->priority[i]; | |
| 97 | + if (s->fiq & (1 << i)) { | |
| 98 | + if (p > p_fiq) { | |
| 99 | + p_fiq = p; | |
| 100 | + intr_fiq = i; | |
| 101 | + } | |
| 102 | + } else { | |
| 103 | + if (p > p_irq) { | |
| 104 | + p_irq = p; | |
| 105 | + intr_irq = i; | |
| 106 | + } | |
| 107 | + } | |
| 108 | + | |
| 109 | + f = ffs(level >> 1); | |
| 110 | + } | |
| 111 | + | |
| 112 | + s->sir_irq = intr_irq; | |
| 113 | + s->sir_fiq = intr_fiq; | |
| 114 | +} | |
| 115 | + | |
| 116 | +#define INT_FALLING_EDGE 0 | |
| 117 | +#define INT_LOW_LEVEL 1 | |
| 118 | + | |
| 119 | +static void omap_set_intr(void *opaque, int irq, int req) | |
| 120 | +{ | |
| 121 | + struct omap_intr_handler_s *ih = (struct omap_intr_handler_s *) opaque; | |
| 122 | + uint32_t rise; | |
| 123 | + | |
| 124 | + if (req) { | |
| 125 | + rise = ~ih->irqs & (1 << irq); | |
| 126 | + ih->irqs |= rise; | |
| 127 | + ih->stats[irq] ++; | |
| 128 | + } else { | |
| 129 | + rise = ih->sens_edge & ih->irqs & (1 << irq); | |
| 130 | + ih->irqs &= ~rise; | |
| 131 | + } | |
| 132 | + | |
| 133 | + if (rise & ~ih->mask) { | |
| 134 | + omap_inth_sir_update(ih); | |
| 135 | + | |
| 136 | + omap_inth_update(ih); | |
| 137 | + } | |
| 138 | +} | |
| 139 | + | |
| 140 | +static uint32_t omap_inth_read(void *opaque, target_phys_addr_t addr) | |
| 141 | +{ | |
| 142 | + struct omap_intr_handler_s *s = (struct omap_intr_handler_s *) opaque; | |
| 143 | + int i, offset = addr - s->base; | |
| 144 | + | |
| 145 | + switch (offset) { | |
| 146 | + case 0x00: /* ITR */ | |
| 147 | + return s->irqs; | |
| 148 | + | |
| 149 | + case 0x04: /* MIR */ | |
| 150 | + return s->mask; | |
| 151 | + | |
| 152 | + case 0x10: /* SIR_IRQ_CODE */ | |
| 153 | + i = s->sir_irq; | |
| 154 | + if (((s->sens_edge >> i) & 1) == INT_FALLING_EDGE && i) { | |
| 155 | + s->irqs &= ~(1 << i); | |
| 156 | + omap_inth_sir_update(s); | |
| 157 | + omap_inth_update(s); | |
| 158 | + } | |
| 159 | + return i; | |
| 160 | + | |
| 161 | + case 0x14: /* SIR_FIQ_CODE */ | |
| 162 | + i = s->sir_fiq; | |
| 163 | + if (((s->sens_edge >> i) & 1) == INT_FALLING_EDGE && i) { | |
| 164 | + s->irqs &= ~(1 << i); | |
| 165 | + omap_inth_sir_update(s); | |
| 166 | + omap_inth_update(s); | |
| 167 | + } | |
| 168 | + return i; | |
| 169 | + | |
| 170 | + case 0x18: /* CONTROL_REG */ | |
| 171 | + return 0; | |
| 172 | + | |
| 173 | + case 0x1c: /* ILR0 */ | |
| 174 | + case 0x20: /* ILR1 */ | |
| 175 | + case 0x24: /* ILR2 */ | |
| 176 | + case 0x28: /* ILR3 */ | |
| 177 | + case 0x2c: /* ILR4 */ | |
| 178 | + case 0x30: /* ILR5 */ | |
| 179 | + case 0x34: /* ILR6 */ | |
| 180 | + case 0x38: /* ILR7 */ | |
| 181 | + case 0x3c: /* ILR8 */ | |
| 182 | + case 0x40: /* ILR9 */ | |
| 183 | + case 0x44: /* ILR10 */ | |
| 184 | + case 0x48: /* ILR11 */ | |
| 185 | + case 0x4c: /* ILR12 */ | |
| 186 | + case 0x50: /* ILR13 */ | |
| 187 | + case 0x54: /* ILR14 */ | |
| 188 | + case 0x58: /* ILR15 */ | |
| 189 | + case 0x5c: /* ILR16 */ | |
| 190 | + case 0x60: /* ILR17 */ | |
| 191 | + case 0x64: /* ILR18 */ | |
| 192 | + case 0x68: /* ILR19 */ | |
| 193 | + case 0x6c: /* ILR20 */ | |
| 194 | + case 0x70: /* ILR21 */ | |
| 195 | + case 0x74: /* ILR22 */ | |
| 196 | + case 0x78: /* ILR23 */ | |
| 197 | + case 0x7c: /* ILR24 */ | |
| 198 | + case 0x80: /* ILR25 */ | |
| 199 | + case 0x84: /* ILR26 */ | |
| 200 | + case 0x88: /* ILR27 */ | |
| 201 | + case 0x8c: /* ILR28 */ | |
| 202 | + case 0x90: /* ILR29 */ | |
| 203 | + case 0x94: /* ILR30 */ | |
| 204 | + case 0x98: /* ILR31 */ | |
| 205 | + i = (offset - 0x1c) >> 2; | |
| 206 | + return (s->priority[i] << 2) | | |
| 207 | + (((s->sens_edge >> i) & 1) << 1) | | |
| 208 | + ((s->fiq >> i) & 1); | |
| 209 | + | |
| 210 | + case 0x9c: /* ISR */ | |
| 211 | + return 0x00000000; | |
| 212 | + | |
| 213 | + default: | |
| 214 | + OMAP_BAD_REG(addr); | |
| 215 | + break; | |
| 216 | + } | |
| 217 | + return 0; | |
| 218 | +} | |
| 219 | + | |
| 220 | +static void omap_inth_write(void *opaque, target_phys_addr_t addr, | |
| 221 | + uint32_t value) | |
| 222 | +{ | |
| 223 | + struct omap_intr_handler_s *s = (struct omap_intr_handler_s *) opaque; | |
| 224 | + int i, offset = addr - s->base; | |
| 225 | + | |
| 226 | + switch (offset) { | |
| 227 | + case 0x00: /* ITR */ | |
| 228 | + s->irqs &= value; | |
| 229 | + omap_inth_sir_update(s); | |
| 230 | + omap_inth_update(s); | |
| 231 | + return; | |
| 232 | + | |
| 233 | + case 0x04: /* MIR */ | |
| 234 | + s->mask = value; | |
| 235 | + omap_inth_sir_update(s); | |
| 236 | + omap_inth_update(s); | |
| 237 | + return; | |
| 238 | + | |
| 239 | + case 0x10: /* SIR_IRQ_CODE */ | |
| 240 | + case 0x14: /* SIR_FIQ_CODE */ | |
| 241 | + OMAP_RO_REG(addr); | |
| 242 | + break; | |
| 243 | + | |
| 244 | + case 0x18: /* CONTROL_REG */ | |
| 245 | + if (value & 2) | |
| 246 | + s->new_fiq_agr = ~0; | |
| 247 | + if (value & 1) | |
| 248 | + s->new_irq_agr = ~0; | |
| 249 | + omap_inth_update(s); | |
| 250 | + return; | |
| 251 | + | |
| 252 | + case 0x1c: /* ILR0 */ | |
| 253 | + case 0x20: /* ILR1 */ | |
| 254 | + case 0x24: /* ILR2 */ | |
| 255 | + case 0x28: /* ILR3 */ | |
| 256 | + case 0x2c: /* ILR4 */ | |
| 257 | + case 0x30: /* ILR5 */ | |
| 258 | + case 0x34: /* ILR6 */ | |
| 259 | + case 0x38: /* ILR7 */ | |
| 260 | + case 0x3c: /* ILR8 */ | |
| 261 | + case 0x40: /* ILR9 */ | |
| 262 | + case 0x44: /* ILR10 */ | |
| 263 | + case 0x48: /* ILR11 */ | |
| 264 | + case 0x4c: /* ILR12 */ | |
| 265 | + case 0x50: /* ILR13 */ | |
| 266 | + case 0x54: /* ILR14 */ | |
| 267 | + case 0x58: /* ILR15 */ | |
| 268 | + case 0x5c: /* ILR16 */ | |
| 269 | + case 0x60: /* ILR17 */ | |
| 270 | + case 0x64: /* ILR18 */ | |
| 271 | + case 0x68: /* ILR19 */ | |
| 272 | + case 0x6c: /* ILR20 */ | |
| 273 | + case 0x70: /* ILR21 */ | |
| 274 | + case 0x74: /* ILR22 */ | |
| 275 | + case 0x78: /* ILR23 */ | |
| 276 | + case 0x7c: /* ILR24 */ | |
| 277 | + case 0x80: /* ILR25 */ | |
| 278 | + case 0x84: /* ILR26 */ | |
| 279 | + case 0x88: /* ILR27 */ | |
| 280 | + case 0x8c: /* ILR28 */ | |
| 281 | + case 0x90: /* ILR29 */ | |
| 282 | + case 0x94: /* ILR30 */ | |
| 283 | + case 0x98: /* ILR31 */ | |
| 284 | + i = (offset - 0x1c) >> 2; | |
| 285 | + s->priority[i] = (value >> 2) & 0x1f; | |
| 286 | + s->sens_edge &= ~(1 << i); | |
| 287 | + s->sens_edge |= ((value >> 1) & 1) << i; | |
| 288 | + s->fiq &= ~(1 << i); | |
| 289 | + s->fiq |= (value & 1) << i; | |
| 290 | + return; | |
| 291 | + | |
| 292 | + case 0x9c: /* ISR */ | |
| 293 | + for (i = 0; i < 32; i ++) | |
| 294 | + if (value & (1 << i)) { | |
| 295 | + omap_set_intr(s, i, 1); | |
| 296 | + return; | |
| 297 | + } | |
| 298 | + return; | |
| 299 | + | |
| 300 | + default: | |
| 301 | + OMAP_BAD_REG(addr); | |
| 302 | + } | |
| 303 | +} | |
| 304 | + | |
| 305 | +static CPUReadMemoryFunc *omap_inth_readfn[] = { | |
| 306 | + omap_badwidth_read32, | |
| 307 | + omap_badwidth_read32, | |
| 308 | + omap_inth_read, | |
| 309 | +}; | |
| 310 | + | |
| 311 | +static CPUWriteMemoryFunc *omap_inth_writefn[] = { | |
| 312 | + omap_inth_write, | |
| 313 | + omap_inth_write, | |
| 314 | + omap_inth_write, | |
| 315 | +}; | |
| 316 | + | |
| 317 | +static void omap_inth_reset(struct omap_intr_handler_s *s) | |
| 318 | +{ | |
| 319 | + s->irqs = 0x00000000; | |
| 320 | + s->mask = 0xffffffff; | |
| 321 | + s->sens_edge = 0x00000000; | |
| 322 | + s->fiq = 0x00000000; | |
| 323 | + memset(s->priority, 0, sizeof(s->priority)); | |
| 324 | + s->new_irq_agr = ~0; | |
| 325 | + s->new_fiq_agr = ~0; | |
| 326 | + s->sir_irq = 0; | |
| 327 | + s->sir_fiq = 0; | |
| 328 | + | |
| 329 | + omap_inth_update(s); | |
| 330 | +} | |
| 331 | + | |
| 332 | +struct omap_intr_handler_s *omap_inth_init(target_phys_addr_t base, | |
| 333 | + unsigned long size, qemu_irq parent[2], omap_clk clk) | |
| 334 | +{ | |
| 335 | + int iomemtype; | |
| 336 | + struct omap_intr_handler_s *s = (struct omap_intr_handler_s *) | |
| 337 | + qemu_mallocz(sizeof(struct omap_intr_handler_s)); | |
| 338 | + | |
| 339 | + s->parent_pic = parent; | |
| 340 | + s->base = base; | |
| 341 | + s->pins = qemu_allocate_irqs(omap_set_intr, s, 32); | |
| 342 | + omap_inth_reset(s); | |
| 343 | + | |
| 344 | + iomemtype = cpu_register_io_memory(0, omap_inth_readfn, | |
| 345 | + omap_inth_writefn, s); | |
| 346 | + cpu_register_physical_memory(s->base, size, iomemtype); | |
| 347 | + | |
| 348 | + return s; | |
| 349 | +} | |
| 350 | + | |
| 351 | +/* OMAP1 DMA module */ | |
| 352 | +typedef enum { | |
| 353 | + constant = 0, | |
| 354 | + post_incremented, | |
| 355 | + single_index, | |
| 356 | + double_index, | |
| 357 | +} omap_dma_addressing_t; | |
| 358 | + | |
| 359 | +struct omap_dma_channel_s { | |
| 360 | + int burst[2]; | |
| 361 | + int pack[2]; | |
| 362 | + enum omap_dma_port port[2]; | |
| 363 | + target_phys_addr_t addr[2]; | |
| 364 | + omap_dma_addressing_t mode[2]; | |
| 365 | + int data_type; | |
| 366 | + int end_prog; | |
| 367 | + int repeat; | |
| 368 | + int auto_init; | |
| 369 | + int priority; | |
| 370 | + int fs; | |
| 371 | + int sync; | |
| 372 | + int running; | |
| 373 | + int interrupts; | |
| 374 | + int status; | |
| 375 | + int signalled; | |
| 376 | + int post_sync; | |
| 377 | + int transfer; | |
| 378 | + uint16_t elements; | |
| 379 | + uint16_t frames; | |
| 380 | + uint16_t frame_index; | |
| 381 | + uint16_t element_index; | |
| 382 | + uint16_t cpc; | |
| 383 | + | |
| 384 | + struct omap_dma_reg_set_s { | |
| 385 | + target_phys_addr_t src, dest; | |
| 386 | + int frame; | |
| 387 | + int element; | |
| 388 | + int frame_delta[2]; | |
| 389 | + int elem_delta[2]; | |
| 390 | + int frames; | |
| 391 | + int elements; | |
| 392 | + } active_set; | |
| 393 | +}; | |
| 394 | + | |
| 395 | +struct omap_dma_s { | |
| 396 | + qemu_irq *ih; | |
| 397 | + QEMUTimer *tm; | |
| 398 | + struct omap_mpu_state_s *mpu; | |
| 399 | + target_phys_addr_t base; | |
| 400 | + omap_clk clk; | |
| 401 | + int64_t delay; | |
| 402 | + | |
| 403 | + uint16_t gcr; | |
| 404 | + int run_count; | |
| 405 | + | |
| 406 | + int chans; | |
| 407 | + struct omap_dma_channel_s ch[16]; | |
| 408 | + struct omap_dma_lcd_channel_s lcd_ch; | |
| 409 | +}; | |
| 410 | + | |
| 411 | +static void omap_dma_interrupts_update(struct omap_dma_s *s) | |
| 412 | +{ | |
| 413 | + /* First three interrupts are shared between two channels each. */ | |
| 414 | + qemu_set_irq(s->ih[OMAP_INT_DMA_CH0_6], | |
| 415 | + (s->ch[0].status | s->ch[6].status) & 0x3f); | |
| 416 | + qemu_set_irq(s->ih[OMAP_INT_DMA_CH1_7], | |
| 417 | + (s->ch[1].status | s->ch[7].status) & 0x3f); | |
| 418 | + qemu_set_irq(s->ih[OMAP_INT_DMA_CH2_8], | |
| 419 | + (s->ch[2].status | s->ch[8].status) & 0x3f); | |
| 420 | + qemu_set_irq(s->ih[OMAP_INT_DMA_CH3], | |
| 421 | + (s->ch[3].status) & 0x3f); | |
| 422 | + qemu_set_irq(s->ih[OMAP_INT_DMA_CH4], | |
| 423 | + (s->ch[4].status) & 0x3f); | |
| 424 | + qemu_set_irq(s->ih[OMAP_INT_DMA_CH5], | |
| 425 | + (s->ch[5].status) & 0x3f); | |
| 426 | +} | |
| 427 | + | |
| 428 | +static void omap_dma_channel_load(struct omap_dma_s *s, int ch) | |
| 429 | +{ | |
| 430 | + struct omap_dma_reg_set_s *a = &s->ch[ch].active_set; | |
| 431 | + int i; | |
| 432 | + | |
| 433 | + /* | |
| 434 | + * TODO: verify address ranges and alignment | |
| 435 | + * TODO: port endianness | |
| 436 | + */ | |
| 437 | + | |
| 438 | + a->src = s->ch[ch].addr[0]; | |
| 439 | + a->dest = s->ch[ch].addr[1]; | |
| 440 | + a->frames = s->ch[ch].frames; | |
| 441 | + a->elements = s->ch[ch].elements; | |
| 442 | + a->frame = 0; | |
| 443 | + a->element = 0; | |
| 444 | + | |
| 445 | + if (unlikely(!s->ch[ch].elements || !s->ch[ch].frames)) { | |
| 446 | + printf("%s: bad DMA request\n", __FUNCTION__); | |
| 447 | + return; | |
| 448 | + } | |
| 449 | + | |
| 450 | + for (i = 0; i < 2; i ++) | |
| 451 | + switch (s->ch[ch].mode[i]) { | |
| 452 | + case constant: | |
| 453 | + a->elem_delta[i] = 0; | |
| 454 | + a->frame_delta[i] = 0; | |
| 455 | + break; | |
| 456 | + case post_incremented: | |
| 457 | + a->elem_delta[i] = s->ch[ch].data_type; | |
| 458 | + a->frame_delta[i] = 0; | |
| 459 | + break; | |
| 460 | + case single_index: | |
| 461 | + a->elem_delta[i] = s->ch[ch].data_type + | |
| 462 | + s->ch[ch].element_index - 1; | |
| 463 | + if (s->ch[ch].element_index > 0x7fff) | |
| 464 | + a->elem_delta[i] -= 0x10000; | |
| 465 | + a->frame_delta[i] = 0; | |
| 466 | + break; | |
| 467 | + case double_index: | |
| 468 | + a->elem_delta[i] = s->ch[ch].data_type + | |
| 469 | + s->ch[ch].element_index - 1; | |
| 470 | + if (s->ch[ch].element_index > 0x7fff) | |
| 471 | + a->elem_delta[i] -= 0x10000; | |
| 472 | + a->frame_delta[i] = s->ch[ch].frame_index - | |
| 473 | + s->ch[ch].element_index; | |
| 474 | + if (s->ch[ch].frame_index > 0x7fff) | |
| 475 | + a->frame_delta[i] -= 0x10000; | |
| 476 | + break; | |
| 477 | + default: | |
| 478 | + break; | |
| 479 | + } | |
| 480 | +} | |
| 481 | + | |
| 482 | +static inline void omap_dma_request_run(struct omap_dma_s *s, | |
| 483 | + int channel, int request) | |
| 484 | +{ | |
| 485 | +next_channel: | |
| 486 | + if (request > 0) | |
| 487 | + for (; channel < 9; channel ++) | |
| 488 | + if (s->ch[channel].sync == request && s->ch[channel].running) | |
| 489 | + break; | |
| 490 | + if (channel >= 9) | |
| 491 | + return; | |
| 492 | + | |
| 493 | + if (s->ch[channel].transfer) { | |
| 494 | + if (request > 0) { | |
| 495 | + s->ch[channel ++].post_sync = request; | |
| 496 | + goto next_channel; | |
| 497 | + } | |
| 498 | + s->ch[channel].status |= 0x02; /* Synchronisation drop */ | |
| 499 | + omap_dma_interrupts_update(s); | |
| 500 | + return; | |
| 501 | + } | |
| 502 | + | |
| 503 | + if (!s->ch[channel].signalled) | |
| 504 | + s->run_count ++; | |
| 505 | + s->ch[channel].signalled = 1; | |
| 506 | + | |
| 507 | + if (request > 0) | |
| 508 | + s->ch[channel].status |= 0x40; /* External request */ | |
| 509 | + | |
| 510 | + if (s->delay) | |
| 511 | + qemu_mod_timer(s->tm, qemu_get_clock(vm_clock) + s->delay); | |
| 512 | + | |
| 513 | + if (request > 0) { | |
| 514 | + channel ++; | |
| 515 | + goto next_channel; | |
| 516 | + } | |
| 517 | +} | |
| 518 | + | |
| 519 | +static inline void omap_dma_request_stop(struct omap_dma_s *s, int channel) | |
| 520 | +{ | |
| 521 | + if (s->ch[channel].signalled) | |
| 522 | + s->run_count --; | |
| 523 | + s->ch[channel].signalled = 0; | |
| 524 | + | |
| 525 | + if (!s->run_count) | |
| 526 | + qemu_del_timer(s->tm); | |
| 527 | +} | |
| 528 | + | |
| 529 | +static void omap_dma_channel_run(struct omap_dma_s *s) | |
| 530 | +{ | |
| 531 | + int ch; | |
| 532 | + uint16_t status; | |
| 533 | + uint8_t value[4]; | |
| 534 | + struct omap_dma_port_if_s *src_p, *dest_p; | |
| 535 | + struct omap_dma_reg_set_s *a; | |
| 536 | + | |
| 537 | + for (ch = 0; ch < 9; ch ++) { | |
| 538 | + a = &s->ch[ch].active_set; | |
| 539 | + | |
| 540 | + src_p = &s->mpu->port[s->ch[ch].port[0]]; | |
| 541 | + dest_p = &s->mpu->port[s->ch[ch].port[1]]; | |
| 542 | + if (s->ch[ch].signalled && (!src_p->addr_valid(s->mpu, a->src) || | |
| 543 | + !dest_p->addr_valid(s->mpu, a->dest))) { | |
| 544 | +#if 0 | |
| 545 | + /* Bus time-out */ | |
| 546 | + if (s->ch[ch].interrupts & 0x01) | |
| 547 | + s->ch[ch].status |= 0x01; | |
| 548 | + omap_dma_request_stop(s, ch); | |
| 549 | + continue; | |
| 550 | +#endif | |
| 551 | + printf("%s: Bus time-out in DMA%i operation\n", __FUNCTION__, ch); | |
| 552 | + } | |
| 553 | + | |
| 554 | + status = s->ch[ch].status; | |
| 555 | + while (status == s->ch[ch].status && s->ch[ch].signalled) { | |
| 556 | + /* Transfer a single element */ | |
| 557 | + s->ch[ch].transfer = 1; | |
| 558 | + cpu_physical_memory_read(a->src, value, s->ch[ch].data_type); | |
| 559 | + cpu_physical_memory_write(a->dest, value, s->ch[ch].data_type); | |
| 560 | + s->ch[ch].transfer = 0; | |
| 561 | + | |
| 562 | + a->src += a->elem_delta[0]; | |
| 563 | + a->dest += a->elem_delta[1]; | |
| 564 | + a->element ++; | |
| 565 | + | |
| 566 | + /* Check interrupt conditions */ | |
| 567 | + if (a->element == a->elements) { | |
| 568 | + a->element = 0; | |
| 569 | + a->src += a->frame_delta[0]; | |
| 570 | + a->dest += a->frame_delta[1]; | |
| 571 | + a->frame ++; | |
| 572 | + | |
| 573 | + if (a->frame == a->frames) { | |
| 574 | + if (!s->ch[ch].repeat || !s->ch[ch].auto_init) | |
| 575 | + s->ch[ch].running = 0; | |
| 576 | + | |
| 577 | + if (s->ch[ch].auto_init && | |
| 578 | + (s->ch[ch].repeat || | |
| 579 | + s->ch[ch].end_prog)) | |
| 580 | + omap_dma_channel_load(s, ch); | |
| 581 | + | |
| 582 | + if (s->ch[ch].interrupts & 0x20) | |
| 583 | + s->ch[ch].status |= 0x20; | |
| 584 | + | |
| 585 | + if (!s->ch[ch].sync) | |
| 586 | + omap_dma_request_stop(s, ch); | |
| 587 | + } | |
| 588 | + | |
| 589 | + if (s->ch[ch].interrupts & 0x08) | |
| 590 | + s->ch[ch].status |= 0x08; | |
| 591 | + | |
| 592 | + if (s->ch[ch].sync && s->ch[ch].fs) { | |
| 593 | + s->ch[ch].status &= ~0x40; | |
| 594 | + omap_dma_request_stop(s, ch); | |
| 595 | + } | |
| 596 | + } | |
| 597 | + | |
| 598 | + if (a->element == 1 && a->frame == a->frames - 1) | |
| 599 | + if (s->ch[ch].interrupts & 0x10) | |
| 600 | + s->ch[ch].status |= 0x10; | |
| 601 | + | |
| 602 | + if (a->element == (a->elements >> 1)) | |
| 603 | + if (s->ch[ch].interrupts & 0x04) | |
| 604 | + s->ch[ch].status |= 0x04; | |
| 605 | + | |
| 606 | + if (s->ch[ch].sync && !s->ch[ch].fs) { | |
| 607 | + s->ch[ch].status &= ~0x40; | |
| 608 | + omap_dma_request_stop(s, ch); | |
| 609 | + } | |
| 610 | + | |
| 611 | + /* | |
| 612 | + * Process requests made while the element was | |
| 613 | + * being transferred. | |
| 614 | + */ | |
| 615 | + if (s->ch[ch].post_sync) { | |
| 616 | + omap_dma_request_run(s, 0, s->ch[ch].post_sync); | |
| 617 | + s->ch[ch].post_sync = 0; | |
| 618 | + } | |
| 619 | + | |
| 620 | +#if 0 | |
| 621 | + break; | |
| 622 | +#endif | |
| 623 | + } | |
| 624 | + | |
| 625 | + s->ch[ch].cpc = a->dest & 0x0000ffff; | |
| 626 | + } | |
| 627 | + | |
| 628 | + omap_dma_interrupts_update(s); | |
| 629 | + if (s->run_count && s->delay) | |
| 630 | + qemu_mod_timer(s->tm, qemu_get_clock(vm_clock) + s->delay); | |
| 631 | +} | |
| 632 | + | |
| 633 | +static int omap_dma_ch_reg_read(struct omap_dma_s *s, | |
| 634 | + int ch, int reg, uint16_t *value) { | |
| 635 | + switch (reg) { | |
| 636 | + case 0x00: /* SYS_DMA_CSDP_CH0 */ | |
| 637 | + *value = (s->ch[ch].burst[1] << 14) | | |
| 638 | + (s->ch[ch].pack[1] << 13) | | |
| 639 | + (s->ch[ch].port[1] << 9) | | |
| 640 | + (s->ch[ch].burst[0] << 7) | | |
| 641 | + (s->ch[ch].pack[0] << 6) | | |
| 642 | + (s->ch[ch].port[0] << 2) | | |
| 643 | + (s->ch[ch].data_type >> 1); | |
| 644 | + break; | |
| 645 | + | |
| 646 | + case 0x02: /* SYS_DMA_CCR_CH0 */ | |
| 647 | + *value = (s->ch[ch].mode[1] << 14) | | |
| 648 | + (s->ch[ch].mode[0] << 12) | | |
| 649 | + (s->ch[ch].end_prog << 11) | | |
| 650 | + (s->ch[ch].repeat << 9) | | |
| 651 | + (s->ch[ch].auto_init << 8) | | |
| 652 | + (s->ch[ch].running << 7) | | |
| 653 | + (s->ch[ch].priority << 6) | | |
| 654 | + (s->ch[ch].fs << 5) | s->ch[ch].sync; | |
| 655 | + break; | |
| 656 | + | |
| 657 | + case 0x04: /* SYS_DMA_CICR_CH0 */ | |
| 658 | + *value = s->ch[ch].interrupts; | |
| 659 | + break; | |
| 660 | + | |
| 661 | + case 0x06: /* SYS_DMA_CSR_CH0 */ | |
| 662 | + /* FIXME: shared CSR for channels sharing the interrupts */ | |
| 663 | + *value = s->ch[ch].status; | |
| 664 | + s->ch[ch].status &= 0x40; | |
| 665 | + omap_dma_interrupts_update(s); | |
| 666 | + break; | |
| 667 | + | |
| 668 | + case 0x08: /* SYS_DMA_CSSA_L_CH0 */ | |
| 669 | + *value = s->ch[ch].addr[0] & 0x0000ffff; | |
| 670 | + break; | |
| 671 | + | |
| 672 | + case 0x0a: /* SYS_DMA_CSSA_U_CH0 */ | |
| 673 | + *value = s->ch[ch].addr[0] >> 16; | |
| 674 | + break; | |
| 675 | + | |
| 676 | + case 0x0c: /* SYS_DMA_CDSA_L_CH0 */ | |
| 677 | + *value = s->ch[ch].addr[1] & 0x0000ffff; | |
| 678 | + break; | |
| 679 | + | |
| 680 | + case 0x0e: /* SYS_DMA_CDSA_U_CH0 */ | |
| 681 | + *value = s->ch[ch].addr[1] >> 16; | |
| 682 | + break; | |
| 683 | + | |
| 684 | + case 0x10: /* SYS_DMA_CEN_CH0 */ | |
| 685 | + *value = s->ch[ch].elements; | |
| 686 | + break; | |
| 687 | + | |
| 688 | + case 0x12: /* SYS_DMA_CFN_CH0 */ | |
| 689 | + *value = s->ch[ch].frames; | |
| 690 | + break; | |
| 691 | + | |
| 692 | + case 0x14: /* SYS_DMA_CFI_CH0 */ | |
| 693 | + *value = s->ch[ch].frame_index; | |
| 694 | + break; | |
| 695 | + | |
| 696 | + case 0x16: /* SYS_DMA_CEI_CH0 */ | |
| 697 | + *value = s->ch[ch].element_index; | |
| 698 | + break; | |
| 699 | + | |
| 700 | + case 0x18: /* SYS_DMA_CPC_CH0 */ | |
| 701 | + *value = s->ch[ch].cpc; | |
| 702 | + break; | |
| 703 | + | |
| 704 | + default: | |
| 705 | + return 1; | |
| 706 | + } | |
| 707 | + return 0; | |
| 708 | +} | |
| 709 | + | |
| 710 | +static int omap_dma_ch_reg_write(struct omap_dma_s *s, | |
| 711 | + int ch, int reg, uint16_t value) { | |
| 712 | + switch (reg) { | |
| 713 | + case 0x00: /* SYS_DMA_CSDP_CH0 */ | |
| 714 | + s->ch[ch].burst[1] = (value & 0xc000) >> 14; | |
| 715 | + s->ch[ch].pack[1] = (value & 0x2000) >> 13; | |
| 716 | + s->ch[ch].port[1] = (enum omap_dma_port) ((value & 0x1e00) >> 9); | |
| 717 | + s->ch[ch].burst[0] = (value & 0x0180) >> 7; | |
| 718 | + s->ch[ch].pack[0] = (value & 0x0040) >> 6; | |
| 719 | + s->ch[ch].port[0] = (enum omap_dma_port) ((value & 0x003c) >> 2); | |
| 720 | + s->ch[ch].data_type = (1 << (value & 3)); | |
| 721 | + if (s->ch[ch].port[0] >= omap_dma_port_last) | |
| 722 | + printf("%s: invalid DMA port %i\n", __FUNCTION__, | |
| 723 | + s->ch[ch].port[0]); | |
| 724 | + if (s->ch[ch].port[1] >= omap_dma_port_last) | |
| 725 | + printf("%s: invalid DMA port %i\n", __FUNCTION__, | |
| 726 | + s->ch[ch].port[1]); | |
| 727 | + if ((value & 3) == 3) | |
| 728 | + printf("%s: bad data_type for DMA channel %i\n", __FUNCTION__, ch); | |
| 729 | + break; | |
| 730 | + | |
| 731 | + case 0x02: /* SYS_DMA_CCR_CH0 */ | |
| 732 | + s->ch[ch].mode[1] = (omap_dma_addressing_t) ((value & 0xc000) >> 14); | |
| 733 | + s->ch[ch].mode[0] = (omap_dma_addressing_t) ((value & 0x3000) >> 12); | |
| 734 | + s->ch[ch].end_prog = (value & 0x0800) >> 11; | |
| 735 | + s->ch[ch].repeat = (value & 0x0200) >> 9; | |
| 736 | + s->ch[ch].auto_init = (value & 0x0100) >> 8; | |
| 737 | + s->ch[ch].priority = (value & 0x0040) >> 6; | |
| 738 | + s->ch[ch].fs = (value & 0x0020) >> 5; | |
| 739 | + s->ch[ch].sync = value & 0x001f; | |
| 740 | + if (value & 0x0080) { | |
| 741 | + if (s->ch[ch].running) { | |
| 742 | + if (!s->ch[ch].signalled && | |
| 743 | + s->ch[ch].auto_init && s->ch[ch].end_prog) | |
| 744 | + omap_dma_channel_load(s, ch); | |
| 745 | + } else { | |
| 746 | + s->ch[ch].running = 1; | |
| 747 | + omap_dma_channel_load(s, ch); | |
| 748 | + } | |
| 749 | + if (!s->ch[ch].sync) | |
| 750 | + omap_dma_request_run(s, ch, 0); | |
| 751 | + } else { | |
| 752 | + s->ch[ch].running = 0; | |
| 753 | + omap_dma_request_stop(s, ch); | |
| 754 | + } | |
| 755 | + break; | |
| 756 | + | |
| 757 | + case 0x04: /* SYS_DMA_CICR_CH0 */ | |
| 758 | + s->ch[ch].interrupts = value & 0x003f; | |
| 759 | + break; | |
| 760 | + | |
| 761 | + case 0x06: /* SYS_DMA_CSR_CH0 */ | |
| 762 | + return 1; | |
| 763 | + | |
| 764 | + case 0x08: /* SYS_DMA_CSSA_L_CH0 */ | |
| 765 | + s->ch[ch].addr[0] &= 0xffff0000; | |
| 766 | + s->ch[ch].addr[0] |= value; | |
| 767 | + break; | |
| 768 | + | |
| 769 | + case 0x0a: /* SYS_DMA_CSSA_U_CH0 */ | |
| 770 | + s->ch[ch].addr[0] &= 0x0000ffff; | |
| 771 | + s->ch[ch].addr[0] |= value << 16; | |
| 772 | + break; | |
| 773 | + | |
| 774 | + case 0x0c: /* SYS_DMA_CDSA_L_CH0 */ | |
| 775 | + s->ch[ch].addr[1] &= 0xffff0000; | |
| 776 | + s->ch[ch].addr[1] |= value; | |
| 777 | + break; | |
| 778 | + | |
| 779 | + case 0x0e: /* SYS_DMA_CDSA_U_CH0 */ | |
| 780 | + s->ch[ch].addr[1] &= 0x0000ffff; | |
| 781 | + s->ch[ch].addr[1] |= value << 16; | |
| 782 | + break; | |
| 783 | + | |
| 784 | + case 0x10: /* SYS_DMA_CEN_CH0 */ | |
| 785 | + s->ch[ch].elements = value & 0xffff; | |
| 786 | + break; | |
| 787 | + | |
| 788 | + case 0x12: /* SYS_DMA_CFN_CH0 */ | |
| 789 | + s->ch[ch].frames = value & 0xffff; | |
| 790 | + break; | |
| 791 | + | |
| 792 | + case 0x14: /* SYS_DMA_CFI_CH0 */ | |
| 793 | + s->ch[ch].frame_index = value & 0xffff; | |
| 794 | + break; | |
| 795 | + | |
| 796 | + case 0x16: /* SYS_DMA_CEI_CH0 */ | |
| 797 | + s->ch[ch].element_index = value & 0xffff; | |
| 798 | + break; | |
| 799 | + | |
| 800 | + case 0x18: /* SYS_DMA_CPC_CH0 */ | |
| 801 | + return 1; | |
| 802 | + | |
| 803 | + default: | |
| 804 | + OMAP_BAD_REG((unsigned long) reg); | |
| 805 | + } | |
| 806 | + return 0; | |
| 807 | +} | |
| 808 | + | |
| 809 | +static uint32_t omap_dma_read(void *opaque, target_phys_addr_t addr) | |
| 810 | +{ | |
| 811 | + struct omap_dma_s *s = (struct omap_dma_s *) opaque; | |
| 812 | + int i, reg, ch, offset = addr - s->base; | |
| 813 | + uint16_t ret; | |
| 814 | + | |
| 815 | + switch (offset) { | |
| 816 | + case 0x000 ... 0x2fe: | |
| 817 | + reg = offset & 0x3f; | |
| 818 | + ch = (offset >> 6) & 0x0f; | |
| 819 | + if (omap_dma_ch_reg_read(s, ch, reg, &ret)) | |
| 820 | + break; | |
| 821 | + return ret; | |
| 822 | + | |
| 823 | + case 0x300: /* SYS_DMA_LCD_CTRL */ | |
| 824 | + i = s->lcd_ch.condition; | |
| 825 | + s->lcd_ch.condition = 0; | |
| 826 | + qemu_irq_lower(s->lcd_ch.irq); | |
| 827 | + return ((s->lcd_ch.src == imif) << 6) | (i << 3) | | |
| 828 | + (s->lcd_ch.interrupts << 1) | s->lcd_ch.dual; | |
| 829 | + | |
| 830 | + case 0x302: /* SYS_DMA_LCD_TOP_F1_L */ | |
| 831 | + return s->lcd_ch.src_f1_top & 0xffff; | |
| 832 | + | |
| 833 | + case 0x304: /* SYS_DMA_LCD_TOP_F1_U */ | |
| 834 | + return s->lcd_ch.src_f1_top >> 16; | |
| 835 | + | |
| 836 | + case 0x306: /* SYS_DMA_LCD_BOT_F1_L */ | |
| 837 | + return s->lcd_ch.src_f1_bottom & 0xffff; | |
| 838 | + | |
| 839 | + case 0x308: /* SYS_DMA_LCD_BOT_F1_U */ | |
| 840 | + return s->lcd_ch.src_f1_bottom >> 16; | |
| 841 | + | |
| 842 | + case 0x30a: /* SYS_DMA_LCD_TOP_F2_L */ | |
| 843 | + return s->lcd_ch.src_f2_top & 0xffff; | |
| 844 | + | |
| 845 | + case 0x30c: /* SYS_DMA_LCD_TOP_F2_U */ | |
| 846 | + return s->lcd_ch.src_f2_top >> 16; | |
| 847 | + | |
| 848 | + case 0x30e: /* SYS_DMA_LCD_BOT_F2_L */ | |
| 849 | + return s->lcd_ch.src_f2_bottom & 0xffff; | |
| 850 | + | |
| 851 | + case 0x310: /* SYS_DMA_LCD_BOT_F2_U */ | |
| 852 | + return s->lcd_ch.src_f2_bottom >> 16; | |
| 853 | + | |
| 854 | + case 0x400: /* SYS_DMA_GCR */ | |
| 855 | + return s->gcr; | |
| 856 | + } | |
| 857 | + | |
| 858 | + OMAP_BAD_REG(addr); | |
| 859 | + return 0; | |
| 860 | +} | |
| 861 | + | |
| 862 | +static void omap_dma_write(void *opaque, target_phys_addr_t addr, | |
| 863 | + uint32_t value) | |
| 864 | +{ | |
| 865 | + struct omap_dma_s *s = (struct omap_dma_s *) opaque; | |
| 866 | + int reg, ch, offset = addr - s->base; | |
| 867 | + | |
| 868 | + switch (offset) { | |
| 869 | + case 0x000 ... 0x2fe: | |
| 870 | + reg = offset & 0x3f; | |
| 871 | + ch = (offset >> 6) & 0x0f; | |
| 872 | + if (omap_dma_ch_reg_write(s, ch, reg, value)) | |
| 873 | + OMAP_RO_REG(addr); | |
| 874 | + break; | |
| 875 | + | |
| 876 | + case 0x300: /* SYS_DMA_LCD_CTRL */ | |
| 877 | + s->lcd_ch.src = (value & 0x40) ? imif : emiff; | |
| 878 | + s->lcd_ch.condition = 0; | |
| 879 | + /* Assume no bus errors and thus no BUS_ERROR irq bits. */ | |
| 880 | + s->lcd_ch.interrupts = (value >> 1) & 1; | |
| 881 | + s->lcd_ch.dual = value & 1; | |
| 882 | + break; | |
| 883 | + | |
| 884 | + case 0x302: /* SYS_DMA_LCD_TOP_F1_L */ | |
| 885 | + s->lcd_ch.src_f1_top &= 0xffff0000; | |
| 886 | + s->lcd_ch.src_f1_top |= 0x0000ffff & value; | |
| 887 | + break; | |
| 888 | + | |
| 889 | + case 0x304: /* SYS_DMA_LCD_TOP_F1_U */ | |
| 890 | + s->lcd_ch.src_f1_top &= 0x0000ffff; | |
| 891 | + s->lcd_ch.src_f1_top |= value << 16; | |
| 892 | + break; | |
| 893 | + | |
| 894 | + case 0x306: /* SYS_DMA_LCD_BOT_F1_L */ | |
| 895 | + s->lcd_ch.src_f1_bottom &= 0xffff0000; | |
| 896 | + s->lcd_ch.src_f1_bottom |= 0x0000ffff & value; | |
| 897 | + break; | |
| 898 | + | |
| 899 | + case 0x308: /* SYS_DMA_LCD_BOT_F1_U */ | |
| 900 | + s->lcd_ch.src_f1_bottom &= 0x0000ffff; | |
| 901 | + s->lcd_ch.src_f1_bottom |= value << 16; | |
| 902 | + break; | |
| 903 | + | |
| 904 | + case 0x30a: /* SYS_DMA_LCD_TOP_F2_L */ | |
| 905 | + s->lcd_ch.src_f2_top &= 0xffff0000; | |
| 906 | + s->lcd_ch.src_f2_top |= 0x0000ffff & value; | |
| 907 | + break; | |
| 908 | + | |
| 909 | + case 0x30c: /* SYS_DMA_LCD_TOP_F2_U */ | |
| 910 | + s->lcd_ch.src_f2_top &= 0x0000ffff; | |
| 911 | + s->lcd_ch.src_f2_top |= value << 16; | |
| 912 | + break; | |
| 913 | + | |
| 914 | + case 0x30e: /* SYS_DMA_LCD_BOT_F2_L */ | |
| 915 | + s->lcd_ch.src_f2_bottom &= 0xffff0000; | |
| 916 | + s->lcd_ch.src_f2_bottom |= 0x0000ffff & value; | |
| 917 | + break; | |
| 918 | + | |
| 919 | + case 0x310: /* SYS_DMA_LCD_BOT_F2_U */ | |
| 920 | + s->lcd_ch.src_f2_bottom &= 0x0000ffff; | |
| 921 | + s->lcd_ch.src_f2_bottom |= value << 16; | |
| 922 | + break; | |
| 923 | + | |
| 924 | + case 0x400: /* SYS_DMA_GCR */ | |
| 925 | + s->gcr = value & 0x000c; | |
| 926 | + break; | |
| 927 | + | |
| 928 | + default: | |
| 929 | + OMAP_BAD_REG(addr); | |
| 930 | + } | |
| 931 | +} | |
| 932 | + | |
| 933 | +static CPUReadMemoryFunc *omap_dma_readfn[] = { | |
| 934 | + omap_badwidth_read16, | |
| 935 | + omap_dma_read, | |
| 936 | + omap_badwidth_read16, | |
| 937 | +}; | |
| 938 | + | |
| 939 | +static CPUWriteMemoryFunc *omap_dma_writefn[] = { | |
| 940 | + omap_badwidth_write16, | |
| 941 | + omap_dma_write, | |
| 942 | + omap_badwidth_write16, | |
| 943 | +}; | |
| 944 | + | |
| 945 | +static void omap_dma_request(void *opaque, int drq, int req) | |
| 946 | +{ | |
| 947 | + struct omap_dma_s *s = (struct omap_dma_s *) opaque; | |
| 948 | + /* All the request pins are edge triggered. */ | |
| 949 | + if (req) | |
| 950 | + omap_dma_request_run(s, 0, drq); | |
| 951 | +} | |
| 952 | + | |
| 953 | +static void omap_dma_clk_update(void *opaque, int line, int on) | |
| 954 | +{ | |
| 955 | + struct omap_dma_s *s = (struct omap_dma_s *) opaque; | |
| 956 | + | |
| 957 | + if (on) { | |
| 958 | + s->delay = ticks_per_sec >> 5; | |
| 959 | + if (s->run_count) | |
| 960 | + qemu_mod_timer(s->tm, qemu_get_clock(vm_clock) + s->delay); | |
| 961 | + } else { | |
| 962 | + s->delay = 0; | |
| 963 | + qemu_del_timer(s->tm); | |
| 964 | + } | |
| 965 | +} | |
| 966 | + | |
| 967 | +static void omap_dma_reset(struct omap_dma_s *s) | |
| 968 | +{ | |
| 969 | + int i; | |
| 970 | + | |
| 971 | + qemu_del_timer(s->tm); | |
| 972 | + s->gcr = 0x0004; | |
| 973 | + s->run_count = 0; | |
| 974 | + s->lcd_ch.src = emiff; | |
| 975 | + s->lcd_ch.condition = 0; | |
| 976 | + s->lcd_ch.interrupts = 0; | |
| 977 | + s->lcd_ch.dual = 0; | |
| 978 | + memset(s->ch, 0, sizeof(s->ch)); | |
| 979 | + for (i = 0; i < s->chans; i ++) | |
| 980 | + s->ch[i].interrupts = 0x0003; | |
| 981 | +} | |
| 982 | + | |
| 983 | +struct omap_dma_s *omap_dma_init(target_phys_addr_t base, | |
| 984 | + qemu_irq pic[], struct omap_mpu_state_s *mpu, omap_clk clk) | |
| 985 | +{ | |
| 986 | + int iomemtype; | |
| 987 | + struct omap_dma_s *s = (struct omap_dma_s *) | |
| 988 | + qemu_mallocz(sizeof(struct omap_dma_s)); | |
| 989 | + | |
| 990 | + s->ih = pic; | |
| 991 | + s->base = base; | |
| 992 | + s->chans = 9; | |
| 993 | + s->mpu = mpu; | |
| 994 | + s->clk = clk; | |
| 995 | + s->lcd_ch.irq = pic[OMAP_INT_DMA_LCD]; | |
| 996 | + s->lcd_ch.mpu = mpu; | |
| 997 | + s->tm = qemu_new_timer(vm_clock, (QEMUTimerCB *) omap_dma_channel_run, s); | |
| 998 | + omap_clk_adduser(s->clk, qemu_allocate_irqs(omap_dma_clk_update, s, 1)[0]); | |
| 999 | + mpu->drq = qemu_allocate_irqs(omap_dma_request, s, 32); | |
| 1000 | + omap_dma_reset(s); | |
| 1001 | + | |
| 1002 | + iomemtype = cpu_register_io_memory(0, omap_dma_readfn, | |
| 1003 | + omap_dma_writefn, s); | |
| 1004 | + cpu_register_physical_memory(s->base, 0x800, iomemtype); | |
| 1005 | + | |
| 1006 | + return s; | |
| 1007 | +} | |
| 1008 | + | |
| 1009 | +/* DMA ports */ | |
| 1010 | +int omap_validate_emiff_addr(struct omap_mpu_state_s *s, | |
| 1011 | + target_phys_addr_t addr) | |
| 1012 | +{ | |
| 1013 | + return addr >= OMAP_EMIFF_BASE && addr < OMAP_EMIFF_BASE + s->sdram_size; | |
| 1014 | +} | |
| 1015 | + | |
| 1016 | +int omap_validate_emifs_addr(struct omap_mpu_state_s *s, | |
| 1017 | + target_phys_addr_t addr) | |
| 1018 | +{ | |
| 1019 | + return addr >= OMAP_EMIFS_BASE && addr < OMAP_EMIFF_BASE; | |
| 1020 | +} | |
| 1021 | + | |
| 1022 | +int omap_validate_imif_addr(struct omap_mpu_state_s *s, | |
| 1023 | + target_phys_addr_t addr) | |
| 1024 | +{ | |
| 1025 | + return addr >= OMAP_IMIF_BASE && addr < OMAP_IMIF_BASE + s->sram_size; | |
| 1026 | +} | |
| 1027 | + | |
| 1028 | +int omap_validate_tipb_addr(struct omap_mpu_state_s *s, | |
| 1029 | + target_phys_addr_t addr) | |
| 1030 | +{ | |
| 1031 | + return addr >= 0xfffb0000 && addr < 0xffff0000; | |
| 1032 | +} | |
| 1033 | + | |
| 1034 | +int omap_validate_local_addr(struct omap_mpu_state_s *s, | |
| 1035 | + target_phys_addr_t addr) | |
| 1036 | +{ | |
| 1037 | + return addr >= OMAP_LOCALBUS_BASE && addr < OMAP_LOCALBUS_BASE + 0x1000000; | |
| 1038 | +} | |
| 1039 | + | |
| 1040 | +int omap_validate_tipb_mpui_addr(struct omap_mpu_state_s *s, | |
| 1041 | + target_phys_addr_t addr) | |
| 1042 | +{ | |
| 1043 | + return addr >= 0xe1010000 && addr < 0xe1020004; | |
| 1044 | +} | |
| 1045 | + | |
| 1046 | +/* MPU OS timers */ | |
| 1047 | +struct omap_mpu_timer_s { | |
| 1048 | + qemu_irq irq; | |
| 1049 | + omap_clk clk; | |
| 1050 | + target_phys_addr_t base; | |
| 1051 | + uint32_t val; | |
| 1052 | + int64_t time; | |
| 1053 | + QEMUTimer *timer; | |
| 1054 | + int64_t rate; | |
| 1055 | + int it_ena; | |
| 1056 | + | |
| 1057 | + int enable; | |
| 1058 | + int ptv; | |
| 1059 | + int ar; | |
| 1060 | + int st; | |
| 1061 | + uint32_t reset_val; | |
| 1062 | +}; | |
| 1063 | + | |
| 1064 | +static inline uint32_t omap_timer_read(struct omap_mpu_timer_s *timer) | |
| 1065 | +{ | |
| 1066 | + uint64_t distance = qemu_get_clock(vm_clock) - timer->time; | |
| 1067 | + | |
| 1068 | + if (timer->st && timer->enable && timer->rate) | |
| 1069 | + return timer->val - muldiv64(distance >> (timer->ptv + 1), | |
| 1070 | + timer->rate, ticks_per_sec); | |
| 1071 | + else | |
| 1072 | + return timer->val; | |
| 1073 | +} | |
| 1074 | + | |
| 1075 | +static inline void omap_timer_sync(struct omap_mpu_timer_s *timer) | |
| 1076 | +{ | |
| 1077 | + timer->val = omap_timer_read(timer); | |
| 1078 | + timer->time = qemu_get_clock(vm_clock); | |
| 1079 | +} | |
| 1080 | + | |
| 1081 | +static inline void omap_timer_update(struct omap_mpu_timer_s *timer) | |
| 1082 | +{ | |
| 1083 | + int64_t expires; | |
| 1084 | + | |
| 1085 | + if (timer->enable && timer->st && timer->rate) { | |
| 1086 | + timer->val = timer->reset_val; /* Should skip this on clk enable */ | |
| 1087 | + expires = timer->time + muldiv64(timer->val << (timer->ptv + 1), | |
| 1088 | + ticks_per_sec, timer->rate); | |
| 1089 | + qemu_mod_timer(timer->timer, expires); | |
| 1090 | + } else | |
| 1091 | + qemu_del_timer(timer->timer); | |
| 1092 | +} | |
| 1093 | + | |
| 1094 | +static void omap_timer_tick(void *opaque) | |
| 1095 | +{ | |
| 1096 | + struct omap_mpu_timer_s *timer = (struct omap_mpu_timer_s *) opaque; | |
| 1097 | + omap_timer_sync(timer); | |
| 1098 | + | |
| 1099 | + if (!timer->ar) { | |
| 1100 | + timer->val = 0; | |
| 1101 | + timer->st = 0; | |
| 1102 | + } | |
| 1103 | + | |
| 1104 | + if (timer->it_ena) | |
| 1105 | + qemu_irq_raise(timer->irq); | |
| 1106 | + omap_timer_update(timer); | |
| 1107 | +} | |
| 1108 | + | |
| 1109 | +static void omap_timer_clk_update(void *opaque, int line, int on) | |
| 1110 | +{ | |
| 1111 | + struct omap_mpu_timer_s *timer = (struct omap_mpu_timer_s *) opaque; | |
| 1112 | + | |
| 1113 | + omap_timer_sync(timer); | |
| 1114 | + timer->rate = on ? omap_clk_getrate(timer->clk) : 0; | |
| 1115 | + omap_timer_update(timer); | |
| 1116 | +} | |
| 1117 | + | |
| 1118 | +static void omap_timer_clk_setup(struct omap_mpu_timer_s *timer) | |
| 1119 | +{ | |
| 1120 | + omap_clk_adduser(timer->clk, | |
| 1121 | + qemu_allocate_irqs(omap_timer_clk_update, timer, 1)[0]); | |
| 1122 | + timer->rate = omap_clk_getrate(timer->clk); | |
| 1123 | +} | |
| 1124 | + | |
| 1125 | +static uint32_t omap_mpu_timer_read(void *opaque, target_phys_addr_t addr) | |
| 1126 | +{ | |
| 1127 | + struct omap_mpu_timer_s *s = (struct omap_mpu_timer_s *) opaque; | |
| 1128 | + int offset = addr - s->base; | |
| 1129 | + | |
| 1130 | + switch (offset) { | |
| 1131 | + case 0x00: /* CNTL_TIMER */ | |
| 1132 | + return (s->enable << 5) | (s->ptv << 2) | (s->ar << 1) | s->st; | |
| 1133 | + | |
| 1134 | + case 0x04: /* LOAD_TIM */ | |
| 1135 | + break; | |
| 1136 | + | |
| 1137 | + case 0x08: /* READ_TIM */ | |
| 1138 | + return omap_timer_read(s); | |
| 1139 | + } | |
| 1140 | + | |
| 1141 | + OMAP_BAD_REG(addr); | |
| 1142 | + return 0; | |
| 1143 | +} | |
| 1144 | + | |
| 1145 | +static void omap_mpu_timer_write(void *opaque, target_phys_addr_t addr, | |
| 1146 | + uint32_t value) | |
| 1147 | +{ | |
| 1148 | + struct omap_mpu_timer_s *s = (struct omap_mpu_timer_s *) opaque; | |
| 1149 | + int offset = addr - s->base; | |
| 1150 | + | |
| 1151 | + switch (offset) { | |
| 1152 | + case 0x00: /* CNTL_TIMER */ | |
| 1153 | + omap_timer_sync(s); | |
| 1154 | + s->enable = (value >> 5) & 1; | |
| 1155 | + s->ptv = (value >> 2) & 7; | |
| 1156 | + s->ar = (value >> 1) & 1; | |
| 1157 | + s->st = value & 1; | |
| 1158 | + omap_timer_update(s); | |
| 1159 | + return; | |
| 1160 | + | |
| 1161 | + case 0x04: /* LOAD_TIM */ | |
| 1162 | + s->reset_val = value; | |
| 1163 | + return; | |
| 1164 | + | |
| 1165 | + case 0x08: /* READ_TIM */ | |
| 1166 | + OMAP_RO_REG(addr); | |
| 1167 | + break; | |
| 1168 | + | |
| 1169 | + default: | |
| 1170 | + OMAP_BAD_REG(addr); | |
| 1171 | + } | |
| 1172 | +} | |
| 1173 | + | |
| 1174 | +static CPUReadMemoryFunc *omap_mpu_timer_readfn[] = { | |
| 1175 | + omap_badwidth_read32, | |
| 1176 | + omap_badwidth_read32, | |
| 1177 | + omap_mpu_timer_read, | |
| 1178 | +}; | |
| 1179 | + | |
| 1180 | +static CPUWriteMemoryFunc *omap_mpu_timer_writefn[] = { | |
| 1181 | + omap_badwidth_write32, | |
| 1182 | + omap_badwidth_write32, | |
| 1183 | + omap_mpu_timer_write, | |
| 1184 | +}; | |
| 1185 | + | |
| 1186 | +static void omap_mpu_timer_reset(struct omap_mpu_timer_s *s) | |
| 1187 | +{ | |
| 1188 | + qemu_del_timer(s->timer); | |
| 1189 | + s->enable = 0; | |
| 1190 | + s->reset_val = 31337; | |
| 1191 | + s->val = 0; | |
| 1192 | + s->ptv = 0; | |
| 1193 | + s->ar = 0; | |
| 1194 | + s->st = 0; | |
| 1195 | + s->it_ena = 1; | |
| 1196 | +} | |
| 1197 | + | |
| 1198 | +struct omap_mpu_timer_s *omap_mpu_timer_init(target_phys_addr_t base, | |
| 1199 | + qemu_irq irq, omap_clk clk) | |
| 1200 | +{ | |
| 1201 | + int iomemtype; | |
| 1202 | + struct omap_mpu_timer_s *s = (struct omap_mpu_timer_s *) | |
| 1203 | + qemu_mallocz(sizeof(struct omap_mpu_timer_s)); | |
| 1204 | + | |
| 1205 | + s->irq = irq; | |
| 1206 | + s->clk = clk; | |
| 1207 | + s->base = base; | |
| 1208 | + s->timer = qemu_new_timer(vm_clock, omap_timer_tick, s); | |
| 1209 | + omap_mpu_timer_reset(s); | |
| 1210 | + omap_timer_clk_setup(s); | |
| 1211 | + | |
| 1212 | + iomemtype = cpu_register_io_memory(0, omap_mpu_timer_readfn, | |
| 1213 | + omap_mpu_timer_writefn, s); | |
| 1214 | + cpu_register_physical_memory(s->base, 0x100, iomemtype); | |
| 1215 | + | |
| 1216 | + return s; | |
| 1217 | +} | |
| 1218 | + | |
| 1219 | +/* Watchdog timer */ | |
| 1220 | +struct omap_watchdog_timer_s { | |
| 1221 | + struct omap_mpu_timer_s timer; | |
| 1222 | + uint8_t last_wr; | |
| 1223 | + int mode; | |
| 1224 | + int free; | |
| 1225 | + int reset; | |
| 1226 | +}; | |
| 1227 | + | |
| 1228 | +static uint32_t omap_wd_timer_read(void *opaque, target_phys_addr_t addr) | |
| 1229 | +{ | |
| 1230 | + struct omap_watchdog_timer_s *s = (struct omap_watchdog_timer_s *) opaque; | |
| 1231 | + int offset = addr - s->timer.base; | |
| 1232 | + | |
| 1233 | + switch (offset) { | |
| 1234 | + case 0x00: /* CNTL_TIMER */ | |
| 1235 | + return (s->timer.ptv << 9) | (s->timer.ar << 8) | | |
| 1236 | + (s->timer.st << 7) | (s->free << 1); | |
| 1237 | + | |
| 1238 | + case 0x04: /* READ_TIMER */ | |
| 1239 | + return omap_timer_read(&s->timer); | |
| 1240 | + | |
| 1241 | + case 0x08: /* TIMER_MODE */ | |
| 1242 | + return s->mode << 15; | |
| 1243 | + } | |
| 1244 | + | |
| 1245 | + OMAP_BAD_REG(addr); | |
| 1246 | + return 0; | |
| 1247 | +} | |
| 1248 | + | |
| 1249 | +static void omap_wd_timer_write(void *opaque, target_phys_addr_t addr, | |
| 1250 | + uint32_t value) | |
| 1251 | +{ | |
| 1252 | + struct omap_watchdog_timer_s *s = (struct omap_watchdog_timer_s *) opaque; | |
| 1253 | + int offset = addr - s->timer.base; | |
| 1254 | + | |
| 1255 | + switch (offset) { | |
| 1256 | + case 0x00: /* CNTL_TIMER */ | |
| 1257 | + omap_timer_sync(&s->timer); | |
| 1258 | + s->timer.ptv = (value >> 9) & 7; | |
| 1259 | + s->timer.ar = (value >> 8) & 1; | |
| 1260 | + s->timer.st = (value >> 7) & 1; | |
| 1261 | + s->free = (value >> 1) & 1; | |
| 1262 | + omap_timer_update(&s->timer); | |
| 1263 | + break; | |
| 1264 | + | |
| 1265 | + case 0x04: /* LOAD_TIMER */ | |
| 1266 | + s->timer.reset_val = value & 0xffff; | |
| 1267 | + break; | |
| 1268 | + | |
| 1269 | + case 0x08: /* TIMER_MODE */ | |
| 1270 | + if (!s->mode && ((value >> 15) & 1)) | |
| 1271 | + omap_clk_get(s->timer.clk); | |
| 1272 | + s->mode |= (value >> 15) & 1; | |
| 1273 | + if (s->last_wr == 0xf5) { | |
| 1274 | + if ((value & 0xff) == 0xa0) { | |
| 1275 | + s->mode = 0; | |
| 1276 | + omap_clk_put(s->timer.clk); | |
| 1277 | + } else { | |
| 1278 | + /* XXX: on T|E hardware somehow this has no effect, | |
| 1279 | + * on Zire 71 it works as specified. */ | |
| 1280 | + s->reset = 1; | |
| 1281 | + qemu_system_reset_request(); | |
| 1282 | + } | |
| 1283 | + } | |
| 1284 | + s->last_wr = value & 0xff; | |
| 1285 | + break; | |
| 1286 | + | |
| 1287 | + default: | |
| 1288 | + OMAP_BAD_REG(addr); | |
| 1289 | + } | |
| 1290 | +} | |
| 1291 | + | |
| 1292 | +static CPUReadMemoryFunc *omap_wd_timer_readfn[] = { | |
| 1293 | + omap_badwidth_read16, | |
| 1294 | + omap_wd_timer_read, | |
| 1295 | + omap_badwidth_read16, | |
| 1296 | +}; | |
| 1297 | + | |
| 1298 | +static CPUWriteMemoryFunc *omap_wd_timer_writefn[] = { | |
| 1299 | + omap_badwidth_write16, | |
| 1300 | + omap_wd_timer_write, | |
| 1301 | + omap_badwidth_write16, | |
| 1302 | +}; | |
| 1303 | + | |
| 1304 | +static void omap_wd_timer_reset(struct omap_watchdog_timer_s *s) | |
| 1305 | +{ | |
| 1306 | + qemu_del_timer(s->timer.timer); | |
| 1307 | + if (!s->mode) | |
| 1308 | + omap_clk_get(s->timer.clk); | |
| 1309 | + s->mode = 1; | |
| 1310 | + s->free = 1; | |
| 1311 | + s->reset = 0; | |
| 1312 | + s->timer.enable = 1; | |
| 1313 | + s->timer.it_ena = 1; | |
| 1314 | + s->timer.reset_val = 0xffff; | |
| 1315 | + s->timer.val = 0; | |
| 1316 | + s->timer.st = 0; | |
| 1317 | + s->timer.ptv = 0; | |
| 1318 | + s->timer.ar = 0; | |
| 1319 | + omap_timer_update(&s->timer); | |
| 1320 | +} | |
| 1321 | + | |
| 1322 | +struct omap_watchdog_timer_s *omap_wd_timer_init(target_phys_addr_t base, | |
| 1323 | + qemu_irq irq, omap_clk clk) | |
| 1324 | +{ | |
| 1325 | + int iomemtype; | |
| 1326 | + struct omap_watchdog_timer_s *s = (struct omap_watchdog_timer_s *) | |
| 1327 | + qemu_mallocz(sizeof(struct omap_watchdog_timer_s)); | |
| 1328 | + | |
| 1329 | + s->timer.irq = irq; | |
| 1330 | + s->timer.clk = clk; | |
| 1331 | + s->timer.base = base; | |
| 1332 | + s->timer.timer = qemu_new_timer(vm_clock, omap_timer_tick, &s->timer); | |
| 1333 | + omap_wd_timer_reset(s); | |
| 1334 | + omap_timer_clk_setup(&s->timer); | |
| 1335 | + | |
| 1336 | + iomemtype = cpu_register_io_memory(0, omap_wd_timer_readfn, | |
| 1337 | + omap_wd_timer_writefn, s); | |
| 1338 | + cpu_register_physical_memory(s->timer.base, 0x100, iomemtype); | |
| 1339 | + | |
| 1340 | + return s; | |
| 1341 | +} | |
| 1342 | + | |
| 1343 | +/* 32-kHz timer */ | |
| 1344 | +struct omap_32khz_timer_s { | |
| 1345 | + struct omap_mpu_timer_s timer; | |
| 1346 | +}; | |
| 1347 | + | |
| 1348 | +static uint32_t omap_os_timer_read(void *opaque, target_phys_addr_t addr) | |
| 1349 | +{ | |
| 1350 | + struct omap_32khz_timer_s *s = (struct omap_32khz_timer_s *) opaque; | |
| 1351 | + int offset = addr - s->timer.base; | |
| 1352 | + | |
| 1353 | + switch (offset) { | |
| 1354 | + case 0x00: /* TVR */ | |
| 1355 | + return s->timer.reset_val; | |
| 1356 | + | |
| 1357 | + case 0x04: /* TCR */ | |
| 1358 | + return omap_timer_read(&s->timer); | |
| 1359 | + | |
| 1360 | + case 0x08: /* CR */ | |
| 1361 | + return (s->timer.ar << 3) | (s->timer.it_ena << 2) | s->timer.st; | |
| 1362 | + | |
| 1363 | + default: | |
| 1364 | + break; | |
| 1365 | + } | |
| 1366 | + OMAP_BAD_REG(addr); | |
| 1367 | + return 0; | |
| 1368 | +} | |
| 1369 | + | |
| 1370 | +static void omap_os_timer_write(void *opaque, target_phys_addr_t addr, | |
| 1371 | + uint32_t value) | |
| 1372 | +{ | |
| 1373 | + struct omap_32khz_timer_s *s = (struct omap_32khz_timer_s *) opaque; | |
| 1374 | + int offset = addr - s->timer.base; | |
| 1375 | + | |
| 1376 | + switch (offset) { | |
| 1377 | + case 0x00: /* TVR */ | |
| 1378 | + s->timer.reset_val = value & 0x00ffffff; | |
| 1379 | + break; | |
| 1380 | + | |
| 1381 | + case 0x04: /* TCR */ | |
| 1382 | + OMAP_RO_REG(addr); | |
| 1383 | + break; | |
| 1384 | + | |
| 1385 | + case 0x08: /* CR */ | |
| 1386 | + s->timer.ar = (value >> 3) & 1; | |
| 1387 | + s->timer.it_ena = (value >> 2) & 1; | |
| 1388 | + if (s->timer.st != (value & 1) || (value & 2)) { | |
| 1389 | + omap_timer_sync(&s->timer); | |
| 1390 | + s->timer.enable = value & 1; | |
| 1391 | + s->timer.st = value & 1; | |
| 1392 | + omap_timer_update(&s->timer); | |
| 1393 | + } | |
| 1394 | + break; | |
| 1395 | + | |
| 1396 | + default: | |
| 1397 | + OMAP_BAD_REG(addr); | |
| 1398 | + } | |
| 1399 | +} | |
| 1400 | + | |
| 1401 | +static CPUReadMemoryFunc *omap_os_timer_readfn[] = { | |
| 1402 | + omap_badwidth_read32, | |
| 1403 | + omap_badwidth_read32, | |
| 1404 | + omap_os_timer_read, | |
| 1405 | +}; | |
| 1406 | + | |
| 1407 | +static CPUWriteMemoryFunc *omap_os_timer_writefn[] = { | |
| 1408 | + omap_badwidth_write32, | |
| 1409 | + omap_badwidth_write32, | |
| 1410 | + omap_os_timer_write, | |
| 1411 | +}; | |
| 1412 | + | |
| 1413 | +static void omap_os_timer_reset(struct omap_32khz_timer_s *s) | |
| 1414 | +{ | |
| 1415 | + qemu_del_timer(s->timer.timer); | |
| 1416 | + s->timer.enable = 0; | |
| 1417 | + s->timer.it_ena = 0; | |
| 1418 | + s->timer.reset_val = 0x00ffffff; | |
| 1419 | + s->timer.val = 0; | |
| 1420 | + s->timer.st = 0; | |
| 1421 | + s->timer.ptv = 0; | |
| 1422 | + s->timer.ar = 1; | |
| 1423 | +} | |
| 1424 | + | |
| 1425 | +struct omap_32khz_timer_s *omap_os_timer_init(target_phys_addr_t base, | |
| 1426 | + qemu_irq irq, omap_clk clk) | |
| 1427 | +{ | |
| 1428 | + int iomemtype; | |
| 1429 | + struct omap_32khz_timer_s *s = (struct omap_32khz_timer_s *) | |
| 1430 | + qemu_mallocz(sizeof(struct omap_32khz_timer_s)); | |
| 1431 | + | |
| 1432 | + s->timer.irq = irq; | |
| 1433 | + s->timer.clk = clk; | |
| 1434 | + s->timer.base = base; | |
| 1435 | + s->timer.timer = qemu_new_timer(vm_clock, omap_timer_tick, &s->timer); | |
| 1436 | + omap_os_timer_reset(s); | |
| 1437 | + omap_timer_clk_setup(&s->timer); | |
| 1438 | + | |
| 1439 | + iomemtype = cpu_register_io_memory(0, omap_os_timer_readfn, | |
| 1440 | + omap_os_timer_writefn, s); | |
| 1441 | + cpu_register_physical_memory(s->timer.base, 0x800, iomemtype); | |
| 1442 | + | |
| 1443 | + return s; | |
| 1444 | +} | |
| 1445 | + | |
| 1446 | +/* Ultra Low-Power Device Module */ | |
| 1447 | +static uint32_t omap_ulpd_pm_read(void *opaque, target_phys_addr_t addr) | |
| 1448 | +{ | |
| 1449 | + struct omap_mpu_state_s *s = (struct omap_mpu_state_s *) opaque; | |
| 1450 | + int offset = addr - s->ulpd_pm_base; | |
| 1451 | + uint16_t ret; | |
| 1452 | + | |
| 1453 | + switch (offset) { | |
| 1454 | + case 0x14: /* IT_STATUS */ | |
| 1455 | + ret = s->ulpd_pm_regs[offset >> 2]; | |
| 1456 | + s->ulpd_pm_regs[offset >> 2] = 0; | |
| 1457 | + qemu_irq_lower(s->irq[1][OMAP_INT_GAUGE_32K]); | |
| 1458 | + return ret; | |
| 1459 | + | |
| 1460 | + case 0x18: /* Reserved */ | |
| 1461 | + case 0x1c: /* Reserved */ | |
| 1462 | + case 0x20: /* Reserved */ | |
| 1463 | + case 0x28: /* Reserved */ | |
| 1464 | + case 0x2c: /* Reserved */ | |
| 1465 | + OMAP_BAD_REG(addr); | |
| 1466 | + case 0x00: /* COUNTER_32_LSB */ | |
| 1467 | + case 0x04: /* COUNTER_32_MSB */ | |
| 1468 | + case 0x08: /* COUNTER_HIGH_FREQ_LSB */ | |
| 1469 | + case 0x0c: /* COUNTER_HIGH_FREQ_MSB */ | |
| 1470 | + case 0x10: /* GAUGING_CTRL */ | |
| 1471 | + case 0x24: /* SETUP_ANALOG_CELL3_ULPD1 */ | |
| 1472 | + case 0x30: /* CLOCK_CTRL */ | |
| 1473 | + case 0x34: /* SOFT_REQ */ | |
| 1474 | + case 0x38: /* COUNTER_32_FIQ */ | |
| 1475 | + case 0x3c: /* DPLL_CTRL */ | |
| 1476 | + case 0x40: /* STATUS_REQ */ | |
| 1477 | + /* XXX: check clk::usecount state for every clock */ | |
| 1478 | + case 0x48: /* LOCL_TIME */ | |
| 1479 | + case 0x4c: /* APLL_CTRL */ | |
| 1480 | + case 0x50: /* POWER_CTRL */ | |
| 1481 | + return s->ulpd_pm_regs[offset >> 2]; | |
| 1482 | + } | |
| 1483 | + | |
| 1484 | + OMAP_BAD_REG(addr); | |
| 1485 | + return 0; | |
| 1486 | +} | |
| 1487 | + | |
| 1488 | +static inline void omap_ulpd_clk_update(struct omap_mpu_state_s *s, | |
| 1489 | + uint16_t diff, uint16_t value) | |
| 1490 | +{ | |
| 1491 | + if (diff & (1 << 4)) /* USB_MCLK_EN */ | |
| 1492 | + omap_clk_onoff(omap_findclk(s, "usb_clk0"), (value >> 4) & 1); | |
| 1493 | + if (diff & (1 << 5)) /* DIS_USB_PVCI_CLK */ | |
| 1494 | + omap_clk_onoff(omap_findclk(s, "usb_w2fc_ck"), (~value >> 5) & 1); | |
| 1495 | +} | |
| 1496 | + | |
| 1497 | +static inline void omap_ulpd_req_update(struct omap_mpu_state_s *s, | |
| 1498 | + uint16_t diff, uint16_t value) | |
| 1499 | +{ | |
| 1500 | + if (diff & (1 << 0)) /* SOFT_DPLL_REQ */ | |
| 1501 | + omap_clk_canidle(omap_findclk(s, "dpll4"), (~value >> 0) & 1); | |
| 1502 | + if (diff & (1 << 1)) /* SOFT_COM_REQ */ | |
| 1503 | + omap_clk_canidle(omap_findclk(s, "com_mclk_out"), (~value >> 1) & 1); | |
| 1504 | + if (diff & (1 << 2)) /* SOFT_SDW_REQ */ | |
| 1505 | + omap_clk_canidle(omap_findclk(s, "bt_mclk_out"), (~value >> 2) & 1); | |
| 1506 | + if (diff & (1 << 3)) /* SOFT_USB_REQ */ | |
| 1507 | + omap_clk_canidle(omap_findclk(s, "usb_clk0"), (~value >> 3) & 1); | |
| 1508 | +} | |
| 1509 | + | |
| 1510 | +static void omap_ulpd_pm_write(void *opaque, target_phys_addr_t addr, | |
| 1511 | + uint32_t value) | |
| 1512 | +{ | |
| 1513 | + struct omap_mpu_state_s *s = (struct omap_mpu_state_s *) opaque; | |
| 1514 | + int offset = addr - s->ulpd_pm_base; | |
| 1515 | + int64_t now, ticks; | |
| 1516 | + int div, mult; | |
| 1517 | + static const int bypass_div[4] = { 1, 2, 4, 4 }; | |
| 1518 | + uint16_t diff; | |
| 1519 | + | |
| 1520 | + switch (offset) { | |
| 1521 | + case 0x00: /* COUNTER_32_LSB */ | |
| 1522 | + case 0x04: /* COUNTER_32_MSB */ | |
| 1523 | + case 0x08: /* COUNTER_HIGH_FREQ_LSB */ | |
| 1524 | + case 0x0c: /* COUNTER_HIGH_FREQ_MSB */ | |
| 1525 | + case 0x14: /* IT_STATUS */ | |
| 1526 | + case 0x40: /* STATUS_REQ */ | |
| 1527 | + OMAP_RO_REG(addr); | |
| 1528 | + break; | |
| 1529 | + | |
| 1530 | + case 0x10: /* GAUGING_CTRL */ | |
| 1531 | + /* Bits 0 and 1 seem to be confused in the OMAP 310 TRM */ | |
| 1532 | + if ((s->ulpd_pm_regs[offset >> 2] ^ value) & 1) { | |
| 1533 | + now = qemu_get_clock(vm_clock); | |
| 1534 | + | |
| 1535 | + if (value & 1) | |
| 1536 | + s->ulpd_gauge_start = now; | |
| 1537 | + else { | |
| 1538 | + now -= s->ulpd_gauge_start; | |
| 1539 | + | |
| 1540 | + /* 32-kHz ticks */ | |
| 1541 | + ticks = muldiv64(now, 32768, ticks_per_sec); | |
| 1542 | + s->ulpd_pm_regs[0x00 >> 2] = (ticks >> 0) & 0xffff; | |
| 1543 | + s->ulpd_pm_regs[0x04 >> 2] = (ticks >> 16) & 0xffff; | |
| 1544 | + if (ticks >> 32) /* OVERFLOW_32K */ | |
| 1545 | + s->ulpd_pm_regs[0x14 >> 2] |= 1 << 2; | |
| 1546 | + | |
| 1547 | + /* High frequency ticks */ | |
| 1548 | + ticks = muldiv64(now, 12000000, ticks_per_sec); | |
| 1549 | + s->ulpd_pm_regs[0x08 >> 2] = (ticks >> 0) & 0xffff; | |
| 1550 | + s->ulpd_pm_regs[0x0c >> 2] = (ticks >> 16) & 0xffff; | |
| 1551 | + if (ticks >> 32) /* OVERFLOW_HI_FREQ */ | |
| 1552 | + s->ulpd_pm_regs[0x14 >> 2] |= 1 << 1; | |
| 1553 | + | |
| 1554 | + s->ulpd_pm_regs[0x14 >> 2] |= 1 << 0; /* IT_GAUGING */ | |
| 1555 | + qemu_irq_raise(s->irq[1][OMAP_INT_GAUGE_32K]); | |
| 1556 | + } | |
| 1557 | + } | |
| 1558 | + s->ulpd_pm_regs[offset >> 2] = value; | |
| 1559 | + break; | |
| 1560 | + | |
| 1561 | + case 0x18: /* Reserved */ | |
| 1562 | + case 0x1c: /* Reserved */ | |
| 1563 | + case 0x20: /* Reserved */ | |
| 1564 | + case 0x28: /* Reserved */ | |
| 1565 | + case 0x2c: /* Reserved */ | |
| 1566 | + OMAP_BAD_REG(addr); | |
| 1567 | + case 0x24: /* SETUP_ANALOG_CELL3_ULPD1 */ | |
| 1568 | + case 0x38: /* COUNTER_32_FIQ */ | |
| 1569 | + case 0x48: /* LOCL_TIME */ | |
| 1570 | + case 0x50: /* POWER_CTRL */ | |
| 1571 | + s->ulpd_pm_regs[offset >> 2] = value; | |
| 1572 | + break; | |
| 1573 | + | |
| 1574 | + case 0x30: /* CLOCK_CTRL */ | |
| 1575 | + diff = s->ulpd_pm_regs[offset >> 2] ^ value; | |
| 1576 | + s->ulpd_pm_regs[offset >> 2] = value & 0x3f; | |
| 1577 | + omap_ulpd_clk_update(s, diff, value); | |
| 1578 | + break; | |
| 1579 | + | |
| 1580 | + case 0x34: /* SOFT_REQ */ | |
| 1581 | + diff = s->ulpd_pm_regs[offset >> 2] ^ value; | |
| 1582 | + s->ulpd_pm_regs[offset >> 2] = value & 0x1f; | |
| 1583 | + omap_ulpd_req_update(s, diff, value); | |
| 1584 | + break; | |
| 1585 | + | |
| 1586 | + case 0x3c: /* DPLL_CTRL */ | |
| 1587 | + /* XXX: OMAP310 TRM claims bit 3 is PLL_ENABLE, and bit 4 is | |
| 1588 | + * omitted altogether, probably a typo. */ | |
| 1589 | + /* This register has identical semantics with DPLL(1:3) control | |
| 1590 | + * registers, see omap_dpll_write() */ | |
| 1591 | + diff = s->ulpd_pm_regs[offset >> 2] & value; | |
| 1592 | + s->ulpd_pm_regs[offset >> 2] = value & 0x2fff; | |
| 1593 | + if (diff & (0x3ff << 2)) { | |
| 1594 | + if (value & (1 << 4)) { /* PLL_ENABLE */ | |
| 1595 | + div = ((value >> 5) & 3) + 1; /* PLL_DIV */ | |
| 1596 | + mult = MIN((value >> 7) & 0x1f, 1); /* PLL_MULT */ | |
| 1597 | + } else { | |
| 1598 | + div = bypass_div[((value >> 2) & 3)]; /* BYPASS_DIV */ | |
| 1599 | + mult = 1; | |
| 1600 | + } | |
| 1601 | + omap_clk_setrate(omap_findclk(s, "dpll4"), div, mult); | |
| 1602 | + } | |
| 1603 | + | |
| 1604 | + /* Enter the desired mode. */ | |
| 1605 | + s->ulpd_pm_regs[offset >> 2] = | |
| 1606 | + (s->ulpd_pm_regs[offset >> 2] & 0xfffe) | | |
| 1607 | + ((s->ulpd_pm_regs[offset >> 2] >> 4) & 1); | |
| 1608 | + | |
| 1609 | + /* Act as if the lock is restored. */ | |
| 1610 | + s->ulpd_pm_regs[offset >> 2] |= 2; | |
| 1611 | + break; | |
| 1612 | + | |
| 1613 | + case 0x4c: /* APLL_CTRL */ | |
| 1614 | + diff = s->ulpd_pm_regs[offset >> 2] & value; | |
| 1615 | + s->ulpd_pm_regs[offset >> 2] = value & 0xf; | |
| 1616 | + if (diff & (1 << 0)) /* APLL_NDPLL_SWITCH */ | |
| 1617 | + omap_clk_reparent(omap_findclk(s, "ck_48m"), omap_findclk(s, | |
| 1618 | + (value & (1 << 0)) ? "apll" : "dpll4")); | |
| 1619 | + break; | |
| 1620 | + | |
| 1621 | + default: | |
| 1622 | + OMAP_BAD_REG(addr); | |
| 1623 | + } | |
| 1624 | +} | |
| 1625 | + | |
| 1626 | +static CPUReadMemoryFunc *omap_ulpd_pm_readfn[] = { | |
| 1627 | + omap_badwidth_read16, | |
| 1628 | + omap_ulpd_pm_read, | |
| 1629 | + omap_badwidth_read16, | |
| 1630 | +}; | |
| 1631 | + | |
| 1632 | +static CPUWriteMemoryFunc *omap_ulpd_pm_writefn[] = { | |
| 1633 | + omap_badwidth_write16, | |
| 1634 | + omap_ulpd_pm_write, | |
| 1635 | + omap_badwidth_write16, | |
| 1636 | +}; | |
| 1637 | + | |
| 1638 | +static void omap_ulpd_pm_reset(struct omap_mpu_state_s *mpu) | |
| 1639 | +{ | |
| 1640 | + mpu->ulpd_pm_regs[0x00 >> 2] = 0x0001; | |
| 1641 | + mpu->ulpd_pm_regs[0x04 >> 2] = 0x0000; | |
| 1642 | + mpu->ulpd_pm_regs[0x08 >> 2] = 0x0001; | |
| 1643 | + mpu->ulpd_pm_regs[0x0c >> 2] = 0x0000; | |
| 1644 | + mpu->ulpd_pm_regs[0x10 >> 2] = 0x0000; | |
| 1645 | + mpu->ulpd_pm_regs[0x18 >> 2] = 0x01; | |
| 1646 | + mpu->ulpd_pm_regs[0x1c >> 2] = 0x01; | |
| 1647 | + mpu->ulpd_pm_regs[0x20 >> 2] = 0x01; | |
| 1648 | + mpu->ulpd_pm_regs[0x24 >> 2] = 0x03ff; | |
| 1649 | + mpu->ulpd_pm_regs[0x28 >> 2] = 0x01; | |
| 1650 | + mpu->ulpd_pm_regs[0x2c >> 2] = 0x01; | |
| 1651 | + omap_ulpd_clk_update(mpu, mpu->ulpd_pm_regs[0x30 >> 2], 0x0000); | |
| 1652 | + mpu->ulpd_pm_regs[0x30 >> 2] = 0x0000; | |
| 1653 | + omap_ulpd_req_update(mpu, mpu->ulpd_pm_regs[0x34 >> 2], 0x0000); | |
| 1654 | + mpu->ulpd_pm_regs[0x34 >> 2] = 0x0000; | |
| 1655 | + mpu->ulpd_pm_regs[0x38 >> 2] = 0x0001; | |
| 1656 | + mpu->ulpd_pm_regs[0x3c >> 2] = 0x2211; | |
| 1657 | + mpu->ulpd_pm_regs[0x40 >> 2] = 0x0000; /* FIXME: dump a real STATUS_REQ */ | |
| 1658 | + mpu->ulpd_pm_regs[0x48 >> 2] = 0x960; | |
| 1659 | + mpu->ulpd_pm_regs[0x4c >> 2] = 0x08; | |
| 1660 | + mpu->ulpd_pm_regs[0x50 >> 2] = 0x08; | |
| 1661 | + omap_clk_setrate(omap_findclk(mpu, "dpll4"), 1, 4); | |
| 1662 | + omap_clk_reparent(omap_findclk(mpu, "ck_48m"), omap_findclk(mpu, "dpll4")); | |
| 1663 | +} | |
| 1664 | + | |
| 1665 | +static void omap_ulpd_pm_init(target_phys_addr_t base, | |
| 1666 | + struct omap_mpu_state_s *mpu) | |
| 1667 | +{ | |
| 1668 | + int iomemtype = cpu_register_io_memory(0, omap_ulpd_pm_readfn, | |
| 1669 | + omap_ulpd_pm_writefn, mpu); | |
| 1670 | + | |
| 1671 | + mpu->ulpd_pm_base = base; | |
| 1672 | + cpu_register_physical_memory(mpu->ulpd_pm_base, 0x800, iomemtype); | |
| 1673 | + omap_ulpd_pm_reset(mpu); | |
| 1674 | +} | |
| 1675 | + | |
| 1676 | +/* OMAP Pin Configuration */ | |
| 1677 | +static uint32_t omap_pin_cfg_read(void *opaque, target_phys_addr_t addr) | |
| 1678 | +{ | |
| 1679 | + struct omap_mpu_state_s *s = (struct omap_mpu_state_s *) opaque; | |
| 1680 | + int offset = addr - s->pin_cfg_base; | |
| 1681 | + | |
| 1682 | + switch (offset) { | |
| 1683 | + case 0x00: /* FUNC_MUX_CTRL_0 */ | |
| 1684 | + case 0x04: /* FUNC_MUX_CTRL_1 */ | |
| 1685 | + case 0x08: /* FUNC_MUX_CTRL_2 */ | |
| 1686 | + return s->func_mux_ctrl[offset >> 2]; | |
| 1687 | + | |
| 1688 | + case 0x0c: /* COMP_MODE_CTRL_0 */ | |
| 1689 | + return s->comp_mode_ctrl[0]; | |
| 1690 | + | |
| 1691 | + case 0x10: /* FUNC_MUX_CTRL_3 */ | |
| 1692 | + case 0x14: /* FUNC_MUX_CTRL_4 */ | |
| 1693 | + case 0x18: /* FUNC_MUX_CTRL_5 */ | |
| 1694 | + case 0x1c: /* FUNC_MUX_CTRL_6 */ | |
| 1695 | + case 0x20: /* FUNC_MUX_CTRL_7 */ | |
| 1696 | + case 0x24: /* FUNC_MUX_CTRL_8 */ | |
| 1697 | + case 0x28: /* FUNC_MUX_CTRL_9 */ | |
| 1698 | + case 0x2c: /* FUNC_MUX_CTRL_A */ | |
| 1699 | + case 0x30: /* FUNC_MUX_CTRL_B */ | |
| 1700 | + case 0x34: /* FUNC_MUX_CTRL_C */ | |
| 1701 | + case 0x38: /* FUNC_MUX_CTRL_D */ | |
| 1702 | + return s->func_mux_ctrl[(offset >> 2) - 1]; | |
| 1703 | + | |
| 1704 | + case 0x40: /* PULL_DWN_CTRL_0 */ | |
| 1705 | + case 0x44: /* PULL_DWN_CTRL_1 */ | |
| 1706 | + case 0x48: /* PULL_DWN_CTRL_2 */ | |
| 1707 | + case 0x4c: /* PULL_DWN_CTRL_3 */ | |
| 1708 | + return s->pull_dwn_ctrl[(offset & 0xf) >> 2]; | |
| 1709 | + | |
| 1710 | + case 0x50: /* GATE_INH_CTRL_0 */ | |
| 1711 | + return s->gate_inh_ctrl[0]; | |
| 1712 | + | |
| 1713 | + case 0x60: /* VOLTAGE_CTRL_0 */ | |
| 1714 | + return s->voltage_ctrl[0]; | |
| 1715 | + | |
| 1716 | + case 0x70: /* TEST_DBG_CTRL_0 */ | |
| 1717 | + return s->test_dbg_ctrl[0]; | |
| 1718 | + | |
| 1719 | + case 0x80: /* MOD_CONF_CTRL_0 */ | |
| 1720 | + return s->mod_conf_ctrl[0]; | |
| 1721 | + } | |
| 1722 | + | |
| 1723 | + OMAP_BAD_REG(addr); | |
| 1724 | + return 0; | |
| 1725 | +} | |
| 1726 | + | |
| 1727 | +static inline void omap_pin_funcmux0_update(struct omap_mpu_state_s *s, | |
| 1728 | + uint32_t diff, uint32_t value) | |
| 1729 | +{ | |
| 1730 | + if (s->compat1509) { | |
| 1731 | + if (diff & (1 << 9)) /* BLUETOOTH */ | |
| 1732 | + omap_clk_onoff(omap_findclk(s, "bt_mclk_out"), | |
| 1733 | + (~value >> 9) & 1); | |
| 1734 | + if (diff & (1 << 7)) /* USB.CLKO */ | |
| 1735 | + omap_clk_onoff(omap_findclk(s, "usb.clko"), | |
| 1736 | + (value >> 7) & 1); | |
| 1737 | + } | |
| 1738 | +} | |
| 1739 | + | |
| 1740 | +static inline void omap_pin_funcmux1_update(struct omap_mpu_state_s *s, | |
| 1741 | + uint32_t diff, uint32_t value) | |
| 1742 | +{ | |
| 1743 | + if (s->compat1509) { | |
| 1744 | + if (diff & (1 << 31)) /* MCBSP3_CLK_HIZ_DI */ | |
| 1745 | + omap_clk_onoff(omap_findclk(s, "mcbsp3.clkx"), | |
| 1746 | + (value >> 31) & 1); | |
| 1747 | + if (diff & (1 << 1)) /* CLK32K */ | |
| 1748 | + omap_clk_onoff(omap_findclk(s, "clk32k_out"), | |
| 1749 | + (~value >> 1) & 1); | |
| 1750 | + } | |
| 1751 | +} | |
| 1752 | + | |
| 1753 | +static inline void omap_pin_modconf1_update(struct omap_mpu_state_s *s, | |
| 1754 | + uint32_t diff, uint32_t value) | |
| 1755 | +{ | |
| 1756 | + if (diff & (1 << 31)) /* CONF_MOD_UART3_CLK_MODE_R */ | |
| 1757 | + omap_clk_reparent(omap_findclk(s, "uart3_ck"), | |
| 1758 | + omap_findclk(s, ((value >> 31) & 1) ? | |
| 1759 | + "ck_48m" : "armper_ck")); | |
| 1760 | + if (diff & (1 << 30)) /* CONF_MOD_UART2_CLK_MODE_R */ | |
| 1761 | + omap_clk_reparent(omap_findclk(s, "uart2_ck"), | |
| 1762 | + omap_findclk(s, ((value >> 30) & 1) ? | |
| 1763 | + "ck_48m" : "armper_ck")); | |
| 1764 | + if (diff & (1 << 29)) /* CONF_MOD_UART1_CLK_MODE_R */ | |
| 1765 | + omap_clk_reparent(omap_findclk(s, "uart1_ck"), | |
| 1766 | + omap_findclk(s, ((value >> 29) & 1) ? | |
| 1767 | + "ck_48m" : "armper_ck")); | |
| 1768 | + if (diff & (1 << 23)) /* CONF_MOD_MMC_SD_CLK_REQ_R */ | |
| 1769 | + omap_clk_reparent(omap_findclk(s, "mmc_ck"), | |
| 1770 | + omap_findclk(s, ((value >> 23) & 1) ? | |
| 1771 | + "ck_48m" : "armper_ck")); | |
| 1772 | + if (diff & (1 << 12)) /* CONF_MOD_COM_MCLK_12_48_S */ | |
| 1773 | + omap_clk_reparent(omap_findclk(s, "com_mclk_out"), | |
| 1774 | + omap_findclk(s, ((value >> 12) & 1) ? | |
| 1775 | + "ck_48m" : "armper_ck")); | |
| 1776 | + if (diff & (1 << 9)) /* CONF_MOD_USB_HOST_HHC_UHO */ | |
| 1777 | + omap_clk_onoff(omap_findclk(s, "usb_hhc_ck"), (value >> 9) & 1); | |
| 1778 | +} | |
| 1779 | + | |
| 1780 | +static void omap_pin_cfg_write(void *opaque, target_phys_addr_t addr, | |
| 1781 | + uint32_t value) | |
| 1782 | +{ | |
| 1783 | + struct omap_mpu_state_s *s = (struct omap_mpu_state_s *) opaque; | |
| 1784 | + int offset = addr - s->pin_cfg_base; | |
| 1785 | + uint32_t diff; | |
| 1786 | + | |
| 1787 | + switch (offset) { | |
| 1788 | + case 0x00: /* FUNC_MUX_CTRL_0 */ | |
| 1789 | + diff = s->func_mux_ctrl[offset >> 2] ^ value; | |
| 1790 | + s->func_mux_ctrl[offset >> 2] = value; | |
| 1791 | + omap_pin_funcmux0_update(s, diff, value); | |
| 1792 | + return; | |
| 1793 | + | |
| 1794 | + case 0x04: /* FUNC_MUX_CTRL_1 */ | |
| 1795 | + diff = s->func_mux_ctrl[offset >> 2] ^ value; | |
| 1796 | + s->func_mux_ctrl[offset >> 2] = value; | |
| 1797 | + omap_pin_funcmux1_update(s, diff, value); | |
| 1798 | + return; | |
| 1799 | + | |
| 1800 | + case 0x08: /* FUNC_MUX_CTRL_2 */ | |
| 1801 | + s->func_mux_ctrl[offset >> 2] = value; | |
| 1802 | + return; | |
| 1803 | + | |
| 1804 | + case 0x0c: /* COMP_MODE_CTRL_0 */ | |
| 1805 | + s->comp_mode_ctrl[0] = value; | |
| 1806 | + s->compat1509 = (value != 0x0000eaef); | |
| 1807 | + omap_pin_funcmux0_update(s, ~0, s->func_mux_ctrl[0]); | |
| 1808 | + omap_pin_funcmux1_update(s, ~0, s->func_mux_ctrl[1]); | |
| 1809 | + return; | |
| 1810 | + | |
| 1811 | + case 0x10: /* FUNC_MUX_CTRL_3 */ | |
| 1812 | + case 0x14: /* FUNC_MUX_CTRL_4 */ | |
| 1813 | + case 0x18: /* FUNC_MUX_CTRL_5 */ | |
| 1814 | + case 0x1c: /* FUNC_MUX_CTRL_6 */ | |
| 1815 | + case 0x20: /* FUNC_MUX_CTRL_7 */ | |
| 1816 | + case 0x24: /* FUNC_MUX_CTRL_8 */ | |
| 1817 | + case 0x28: /* FUNC_MUX_CTRL_9 */ | |
| 1818 | + case 0x2c: /* FUNC_MUX_CTRL_A */ | |
| 1819 | + case 0x30: /* FUNC_MUX_CTRL_B */ | |
| 1820 | + case 0x34: /* FUNC_MUX_CTRL_C */ | |
| 1821 | + case 0x38: /* FUNC_MUX_CTRL_D */ | |
| 1822 | + s->func_mux_ctrl[(offset >> 2) - 1] = value; | |
| 1823 | + return; | |
| 1824 | + | |
| 1825 | + case 0x40: /* PULL_DWN_CTRL_0 */ | |
| 1826 | + case 0x44: /* PULL_DWN_CTRL_1 */ | |
| 1827 | + case 0x48: /* PULL_DWN_CTRL_2 */ | |
| 1828 | + case 0x4c: /* PULL_DWN_CTRL_3 */ | |
| 1829 | + s->pull_dwn_ctrl[(offset & 0xf) >> 2] = value; | |
| 1830 | + return; | |
| 1831 | + | |
| 1832 | + case 0x50: /* GATE_INH_CTRL_0 */ | |
| 1833 | + s->gate_inh_ctrl[0] = value; | |
| 1834 | + return; | |
| 1835 | + | |
| 1836 | + case 0x60: /* VOLTAGE_CTRL_0 */ | |
| 1837 | + s->voltage_ctrl[0] = value; | |
| 1838 | + return; | |
| 1839 | + | |
| 1840 | + case 0x70: /* TEST_DBG_CTRL_0 */ | |
| 1841 | + s->test_dbg_ctrl[0] = value; | |
| 1842 | + return; | |
| 1843 | + | |
| 1844 | + case 0x80: /* MOD_CONF_CTRL_0 */ | |
| 1845 | + diff = s->mod_conf_ctrl[0] ^ value; | |
| 1846 | + s->mod_conf_ctrl[0] = value; | |
| 1847 | + omap_pin_modconf1_update(s, diff, value); | |
| 1848 | + return; | |
| 1849 | + | |
| 1850 | + default: | |
| 1851 | + OMAP_BAD_REG(addr); | |
| 1852 | + } | |
| 1853 | +} | |
| 1854 | + | |
| 1855 | +static CPUReadMemoryFunc *omap_pin_cfg_readfn[] = { | |
| 1856 | + omap_badwidth_read32, | |
| 1857 | + omap_badwidth_read32, | |
| 1858 | + omap_pin_cfg_read, | |
| 1859 | +}; | |
| 1860 | + | |
| 1861 | +static CPUWriteMemoryFunc *omap_pin_cfg_writefn[] = { | |
| 1862 | + omap_badwidth_write32, | |
| 1863 | + omap_badwidth_write32, | |
| 1864 | + omap_pin_cfg_write, | |
| 1865 | +}; | |
| 1866 | + | |
| 1867 | +static void omap_pin_cfg_reset(struct omap_mpu_state_s *mpu) | |
| 1868 | +{ | |
| 1869 | + /* Start in Compatibility Mode. */ | |
| 1870 | + mpu->compat1509 = 1; | |
| 1871 | + omap_pin_funcmux0_update(mpu, mpu->func_mux_ctrl[0], 0); | |
| 1872 | + omap_pin_funcmux1_update(mpu, mpu->func_mux_ctrl[1], 0); | |
| 1873 | + omap_pin_modconf1_update(mpu, mpu->mod_conf_ctrl[0], 0); | |
| 1874 | + memset(mpu->func_mux_ctrl, 0, sizeof(mpu->func_mux_ctrl)); | |
| 1875 | + memset(mpu->comp_mode_ctrl, 0, sizeof(mpu->comp_mode_ctrl)); | |
| 1876 | + memset(mpu->pull_dwn_ctrl, 0, sizeof(mpu->pull_dwn_ctrl)); | |
| 1877 | + memset(mpu->gate_inh_ctrl, 0, sizeof(mpu->gate_inh_ctrl)); | |
| 1878 | + memset(mpu->voltage_ctrl, 0, sizeof(mpu->voltage_ctrl)); | |
| 1879 | + memset(mpu->test_dbg_ctrl, 0, sizeof(mpu->test_dbg_ctrl)); | |
| 1880 | + memset(mpu->mod_conf_ctrl, 0, sizeof(mpu->mod_conf_ctrl)); | |
| 1881 | +} | |
| 1882 | + | |
| 1883 | +static void omap_pin_cfg_init(target_phys_addr_t base, | |
| 1884 | + struct omap_mpu_state_s *mpu) | |
| 1885 | +{ | |
| 1886 | + int iomemtype = cpu_register_io_memory(0, omap_pin_cfg_readfn, | |
| 1887 | + omap_pin_cfg_writefn, mpu); | |
| 1888 | + | |
| 1889 | + mpu->pin_cfg_base = base; | |
| 1890 | + cpu_register_physical_memory(mpu->pin_cfg_base, 0x800, iomemtype); | |
| 1891 | + omap_pin_cfg_reset(mpu); | |
| 1892 | +} | |
| 1893 | + | |
| 1894 | +/* Device Identification, Die Identification */ | |
| 1895 | +static uint32_t omap_id_read(void *opaque, target_phys_addr_t addr) | |
| 1896 | +{ | |
| 1897 | + struct omap_mpu_state_s *s = (struct omap_mpu_state_s *) opaque; | |
| 1898 | + | |
| 1899 | + switch (addr) { | |
| 1900 | + case 0xfffe1800: /* DIE_ID_LSB */ | |
| 1901 | + return 0xc9581f0e; | |
| 1902 | + case 0xfffe1804: /* DIE_ID_MSB */ | |
| 1903 | + return 0xa8858bfa; | |
| 1904 | + | |
| 1905 | + case 0xfffe2000: /* PRODUCT_ID_LSB */ | |
| 1906 | + return 0x00aaaafc; | |
| 1907 | + case 0xfffe2004: /* PRODUCT_ID_MSB */ | |
| 1908 | + return 0xcafeb574; | |
| 1909 | + | |
| 1910 | + case 0xfffed400: /* JTAG_ID_LSB */ | |
| 1911 | + switch (s->mpu_model) { | |
| 1912 | + case omap310: | |
| 1913 | + return 0x03310315; | |
| 1914 | + case omap1510: | |
| 1915 | + return 0x03310115; | |
| 1916 | + } | |
| 1917 | + break; | |
| 1918 | + | |
| 1919 | + case 0xfffed404: /* JTAG_ID_MSB */ | |
| 1920 | + switch (s->mpu_model) { | |
| 1921 | + case omap310: | |
| 1922 | + return 0xfb57402f; | |
| 1923 | + case omap1510: | |
| 1924 | + return 0xfb47002f; | |
| 1925 | + } | |
| 1926 | + break; | |
| 1927 | + } | |
| 1928 | + | |
| 1929 | + OMAP_BAD_REG(addr); | |
| 1930 | + return 0; | |
| 1931 | +} | |
| 1932 | + | |
| 1933 | +static void omap_id_write(void *opaque, target_phys_addr_t addr, | |
| 1934 | + uint32_t value) | |
| 1935 | +{ | |
| 1936 | + OMAP_BAD_REG(addr); | |
| 1937 | +} | |
| 1938 | + | |
| 1939 | +static CPUReadMemoryFunc *omap_id_readfn[] = { | |
| 1940 | + omap_badwidth_read32, | |
| 1941 | + omap_badwidth_read32, | |
| 1942 | + omap_id_read, | |
| 1943 | +}; | |
| 1944 | + | |
| 1945 | +static CPUWriteMemoryFunc *omap_id_writefn[] = { | |
| 1946 | + omap_badwidth_write32, | |
| 1947 | + omap_badwidth_write32, | |
| 1948 | + omap_id_write, | |
| 1949 | +}; | |
| 1950 | + | |
| 1951 | +static void omap_id_init(struct omap_mpu_state_s *mpu) | |
| 1952 | +{ | |
| 1953 | + int iomemtype = cpu_register_io_memory(0, omap_id_readfn, | |
| 1954 | + omap_id_writefn, mpu); | |
| 1955 | + cpu_register_physical_memory(0xfffe1800, 0x800, iomemtype); | |
| 1956 | + cpu_register_physical_memory(0xfffed400, 0x100, iomemtype); | |
| 1957 | + if (!cpu_is_omap15xx(mpu)) | |
| 1958 | + cpu_register_physical_memory(0xfffe2000, 0x800, iomemtype); | |
| 1959 | +} | |
| 1960 | + | |
| 1961 | +/* MPUI Control (Dummy) */ | |
| 1962 | +static uint32_t omap_mpui_read(void *opaque, target_phys_addr_t addr) | |
| 1963 | +{ | |
| 1964 | + struct omap_mpu_state_s *s = (struct omap_mpu_state_s *) opaque; | |
| 1965 | + int offset = addr - s->mpui_base; | |
| 1966 | + | |
| 1967 | + switch (offset) { | |
| 1968 | + case 0x00: /* CTRL */ | |
| 1969 | + return s->mpui_ctrl; | |
| 1970 | + case 0x04: /* DEBUG_ADDR */ | |
| 1971 | + return 0x01ffffff; | |
| 1972 | + case 0x08: /* DEBUG_DATA */ | |
| 1973 | + return 0xffffffff; | |
| 1974 | + case 0x0c: /* DEBUG_FLAG */ | |
| 1975 | + return 0x00000800; | |
| 1976 | + case 0x10: /* STATUS */ | |
| 1977 | + return 0x00000000; | |
| 1978 | + | |
| 1979 | + /* Not in OMAP310 */ | |
| 1980 | + case 0x14: /* DSP_STATUS */ | |
| 1981 | + case 0x18: /* DSP_BOOT_CONFIG */ | |
| 1982 | + return 0x00000000; | |
| 1983 | + case 0x1c: /* DSP_MPUI_CONFIG */ | |
| 1984 | + return 0x0000ffff; | |
| 1985 | + } | |
| 1986 | + | |
| 1987 | + OMAP_BAD_REG(addr); | |
| 1988 | + return 0; | |
| 1989 | +} | |
| 1990 | + | |
| 1991 | +static void omap_mpui_write(void *opaque, target_phys_addr_t addr, | |
| 1992 | + uint32_t value) | |
| 1993 | +{ | |
| 1994 | + struct omap_mpu_state_s *s = (struct omap_mpu_state_s *) opaque; | |
| 1995 | + int offset = addr - s->mpui_base; | |
| 1996 | + | |
| 1997 | + switch (offset) { | |
| 1998 | + case 0x00: /* CTRL */ | |
| 1999 | + s->mpui_ctrl = value & 0x007fffff; | |
| 2000 | + break; | |
| 2001 | + | |
| 2002 | + case 0x04: /* DEBUG_ADDR */ | |
| 2003 | + case 0x08: /* DEBUG_DATA */ | |
| 2004 | + case 0x0c: /* DEBUG_FLAG */ | |
| 2005 | + case 0x10: /* STATUS */ | |
| 2006 | + /* Not in OMAP310 */ | |
| 2007 | + case 0x14: /* DSP_STATUS */ | |
| 2008 | + OMAP_RO_REG(addr); | |
| 2009 | + case 0x18: /* DSP_BOOT_CONFIG */ | |
| 2010 | + case 0x1c: /* DSP_MPUI_CONFIG */ | |
| 2011 | + break; | |
| 2012 | + | |
| 2013 | + default: | |
| 2014 | + OMAP_BAD_REG(addr); | |
| 2015 | + } | |
| 2016 | +} | |
| 2017 | + | |
| 2018 | +static CPUReadMemoryFunc *omap_mpui_readfn[] = { | |
| 2019 | + omap_badwidth_read32, | |
| 2020 | + omap_badwidth_read32, | |
| 2021 | + omap_mpui_read, | |
| 2022 | +}; | |
| 2023 | + | |
| 2024 | +static CPUWriteMemoryFunc *omap_mpui_writefn[] = { | |
| 2025 | + omap_badwidth_write32, | |
| 2026 | + omap_badwidth_write32, | |
| 2027 | + omap_mpui_write, | |
| 2028 | +}; | |
| 2029 | + | |
| 2030 | +static void omap_mpui_reset(struct omap_mpu_state_s *s) | |
| 2031 | +{ | |
| 2032 | + s->mpui_ctrl = 0x0003ff1b; | |
| 2033 | +} | |
| 2034 | + | |
| 2035 | +static void omap_mpui_init(target_phys_addr_t base, | |
| 2036 | + struct omap_mpu_state_s *mpu) | |
| 2037 | +{ | |
| 2038 | + int iomemtype = cpu_register_io_memory(0, omap_mpui_readfn, | |
| 2039 | + omap_mpui_writefn, mpu); | |
| 2040 | + | |
| 2041 | + mpu->mpui_base = base; | |
| 2042 | + cpu_register_physical_memory(mpu->mpui_base, 0x100, iomemtype); | |
| 2043 | + | |
| 2044 | + omap_mpui_reset(mpu); | |
| 2045 | +} | |
| 2046 | + | |
| 2047 | +/* TIPB Bridges */ | |
| 2048 | +struct omap_tipb_bridge_s { | |
| 2049 | + target_phys_addr_t base; | |
| 2050 | + qemu_irq abort; | |
| 2051 | + | |
| 2052 | + int width_intr; | |
| 2053 | + uint16_t control; | |
| 2054 | + uint16_t alloc; | |
| 2055 | + uint16_t buffer; | |
| 2056 | + uint16_t enh_control; | |
| 2057 | +}; | |
| 2058 | + | |
| 2059 | +static uint32_t omap_tipb_bridge_read(void *opaque, target_phys_addr_t addr) | |
| 2060 | +{ | |
| 2061 | + struct omap_tipb_bridge_s *s = (struct omap_tipb_bridge_s *) opaque; | |
| 2062 | + int offset = addr - s->base; | |
| 2063 | + | |
| 2064 | + switch (offset) { | |
| 2065 | + case 0x00: /* TIPB_CNTL */ | |
| 2066 | + return s->control; | |
| 2067 | + case 0x04: /* TIPB_BUS_ALLOC */ | |
| 2068 | + return s->alloc; | |
| 2069 | + case 0x08: /* MPU_TIPB_CNTL */ | |
| 2070 | + return s->buffer; | |
| 2071 | + case 0x0c: /* ENHANCED_TIPB_CNTL */ | |
| 2072 | + return s->enh_control; | |
| 2073 | + case 0x10: /* ADDRESS_DBG */ | |
| 2074 | + case 0x14: /* DATA_DEBUG_LOW */ | |
| 2075 | + case 0x18: /* DATA_DEBUG_HIGH */ | |
| 2076 | + return 0xffff; | |
| 2077 | + case 0x1c: /* DEBUG_CNTR_SIG */ | |
| 2078 | + return 0x00f8; | |
| 2079 | + } | |
| 2080 | + | |
| 2081 | + OMAP_BAD_REG(addr); | |
| 2082 | + return 0; | |
| 2083 | +} | |
| 2084 | + | |
| 2085 | +static void omap_tipb_bridge_write(void *opaque, target_phys_addr_t addr, | |
| 2086 | + uint32_t value) | |
| 2087 | +{ | |
| 2088 | + struct omap_tipb_bridge_s *s = (struct omap_tipb_bridge_s *) opaque; | |
| 2089 | + int offset = addr - s->base; | |
| 2090 | + | |
| 2091 | + switch (offset) { | |
| 2092 | + case 0x00: /* TIPB_CNTL */ | |
| 2093 | + s->control = value & 0xffff; | |
| 2094 | + break; | |
| 2095 | + | |
| 2096 | + case 0x04: /* TIPB_BUS_ALLOC */ | |
| 2097 | + s->alloc = value & 0x003f; | |
| 2098 | + break; | |
| 2099 | + | |
| 2100 | + case 0x08: /* MPU_TIPB_CNTL */ | |
| 2101 | + s->buffer = value & 0x0003; | |
| 2102 | + break; | |
| 2103 | + | |
| 2104 | + case 0x0c: /* ENHANCED_TIPB_CNTL */ | |
| 2105 | + s->width_intr = !(value & 2); | |
| 2106 | + s->enh_control = value & 0x000f; | |
| 2107 | + break; | |
| 2108 | + | |
| 2109 | + case 0x10: /* ADDRESS_DBG */ | |
| 2110 | + case 0x14: /* DATA_DEBUG_LOW */ | |
| 2111 | + case 0x18: /* DATA_DEBUG_HIGH */ | |
| 2112 | + case 0x1c: /* DEBUG_CNTR_SIG */ | |
| 2113 | + OMAP_RO_REG(addr); | |
| 2114 | + break; | |
| 2115 | + | |
| 2116 | + default: | |
| 2117 | + OMAP_BAD_REG(addr); | |
| 2118 | + } | |
| 2119 | +} | |
| 2120 | + | |
| 2121 | +static CPUReadMemoryFunc *omap_tipb_bridge_readfn[] = { | |
| 2122 | + omap_badwidth_read16, | |
| 2123 | + omap_tipb_bridge_read, | |
| 2124 | + omap_tipb_bridge_read, | |
| 2125 | +}; | |
| 2126 | + | |
| 2127 | +static CPUWriteMemoryFunc *omap_tipb_bridge_writefn[] = { | |
| 2128 | + omap_badwidth_write16, | |
| 2129 | + omap_tipb_bridge_write, | |
| 2130 | + omap_tipb_bridge_write, | |
| 2131 | +}; | |
| 2132 | + | |
| 2133 | +static void omap_tipb_bridge_reset(struct omap_tipb_bridge_s *s) | |
| 2134 | +{ | |
| 2135 | + s->control = 0xffff; | |
| 2136 | + s->alloc = 0x0009; | |
| 2137 | + s->buffer = 0x0000; | |
| 2138 | + s->enh_control = 0x000f; | |
| 2139 | +} | |
| 2140 | + | |
| 2141 | +struct omap_tipb_bridge_s *omap_tipb_bridge_init(target_phys_addr_t base, | |
| 2142 | + qemu_irq abort_irq, omap_clk clk) | |
| 2143 | +{ | |
| 2144 | + int iomemtype; | |
| 2145 | + struct omap_tipb_bridge_s *s = (struct omap_tipb_bridge_s *) | |
| 2146 | + qemu_mallocz(sizeof(struct omap_tipb_bridge_s)); | |
| 2147 | + | |
| 2148 | + s->abort = abort_irq; | |
| 2149 | + s->base = base; | |
| 2150 | + omap_tipb_bridge_reset(s); | |
| 2151 | + | |
| 2152 | + iomemtype = cpu_register_io_memory(0, omap_tipb_bridge_readfn, | |
| 2153 | + omap_tipb_bridge_writefn, s); | |
| 2154 | + cpu_register_physical_memory(s->base, 0x100, iomemtype); | |
| 2155 | + | |
| 2156 | + return s; | |
| 2157 | +} | |
| 2158 | + | |
| 2159 | +/* Dummy Traffic Controller's Memory Interface */ | |
| 2160 | +static uint32_t omap_tcmi_read(void *opaque, target_phys_addr_t addr) | |
| 2161 | +{ | |
| 2162 | + struct omap_mpu_state_s *s = (struct omap_mpu_state_s *) opaque; | |
| 2163 | + int offset = addr - s->tcmi_base; | |
| 2164 | + uint32_t ret; | |
| 2165 | + | |
| 2166 | + switch (offset) { | |
| 2167 | + case 0xfffecc00: /* IMIF_PRIO */ | |
| 2168 | + case 0xfffecc04: /* EMIFS_PRIO */ | |
| 2169 | + case 0xfffecc08: /* EMIFF_PRIO */ | |
| 2170 | + case 0xfffecc0c: /* EMIFS_CONFIG */ | |
| 2171 | + case 0xfffecc10: /* EMIFS_CS0_CONFIG */ | |
| 2172 | + case 0xfffecc14: /* EMIFS_CS1_CONFIG */ | |
| 2173 | + case 0xfffecc18: /* EMIFS_CS2_CONFIG */ | |
| 2174 | + case 0xfffecc1c: /* EMIFS_CS3_CONFIG */ | |
| 2175 | + case 0xfffecc24: /* EMIFF_MRS */ | |
| 2176 | + case 0xfffecc28: /* TIMEOUT1 */ | |
| 2177 | + case 0xfffecc2c: /* TIMEOUT2 */ | |
| 2178 | + case 0xfffecc30: /* TIMEOUT3 */ | |
| 2179 | + case 0xfffecc3c: /* EMIFF_SDRAM_CONFIG_2 */ | |
| 2180 | + case 0xfffecc40: /* EMIFS_CFG_DYN_WAIT */ | |
| 2181 | + return s->tcmi_regs[offset >> 2]; | |
| 2182 | + | |
| 2183 | + case 0xfffecc20: /* EMIFF_SDRAM_CONFIG */ | |
| 2184 | + ret = s->tcmi_regs[offset >> 2]; | |
| 2185 | + s->tcmi_regs[offset >> 2] &= ~1; /* XXX: Clear SLRF on SDRAM access */ | |
| 2186 | + /* XXX: We can try using the VGA_DIRTY flag for this */ | |
| 2187 | + return ret; | |
| 2188 | + } | |
| 2189 | + | |
| 2190 | + OMAP_BAD_REG(addr); | |
| 2191 | + return 0; | |
| 2192 | +} | |
| 2193 | + | |
| 2194 | +static void omap_tcmi_write(void *opaque, target_phys_addr_t addr, | |
| 2195 | + uint32_t value) | |
| 2196 | +{ | |
| 2197 | + struct omap_mpu_state_s *s = (struct omap_mpu_state_s *) opaque; | |
| 2198 | + int offset = addr - s->tcmi_base; | |
| 2199 | + | |
| 2200 | + switch (offset) { | |
| 2201 | + case 0xfffecc00: /* IMIF_PRIO */ | |
| 2202 | + case 0xfffecc04: /* EMIFS_PRIO */ | |
| 2203 | + case 0xfffecc08: /* EMIFF_PRIO */ | |
| 2204 | + case 0xfffecc10: /* EMIFS_CS0_CONFIG */ | |
| 2205 | + case 0xfffecc14: /* EMIFS_CS1_CONFIG */ | |
| 2206 | + case 0xfffecc18: /* EMIFS_CS2_CONFIG */ | |
| 2207 | + case 0xfffecc1c: /* EMIFS_CS3_CONFIG */ | |
| 2208 | + case 0xfffecc20: /* EMIFF_SDRAM_CONFIG */ | |
| 2209 | + case 0xfffecc24: /* EMIFF_MRS */ | |
| 2210 | + case 0xfffecc28: /* TIMEOUT1 */ | |
| 2211 | + case 0xfffecc2c: /* TIMEOUT2 */ | |
| 2212 | + case 0xfffecc30: /* TIMEOUT3 */ | |
| 2213 | + case 0xfffecc3c: /* EMIFF_SDRAM_CONFIG_2 */ | |
| 2214 | + case 0xfffecc40: /* EMIFS_CFG_DYN_WAIT */ | |
| 2215 | + s->tcmi_regs[offset >> 2] = value; | |
| 2216 | + break; | |
| 2217 | + case 0xfffecc0c: /* EMIFS_CONFIG */ | |
| 2218 | + s->tcmi_regs[offset >> 2] = (value & 0xf) | (1 << 4); | |
| 2219 | + break; | |
| 2220 | + | |
| 2221 | + default: | |
| 2222 | + OMAP_BAD_REG(addr); | |
| 2223 | + } | |
| 2224 | +} | |
| 2225 | + | |
| 2226 | +static CPUReadMemoryFunc *omap_tcmi_readfn[] = { | |
| 2227 | + omap_badwidth_read32, | |
| 2228 | + omap_badwidth_read32, | |
| 2229 | + omap_tcmi_read, | |
| 2230 | +}; | |
| 2231 | + | |
| 2232 | +static CPUWriteMemoryFunc *omap_tcmi_writefn[] = { | |
| 2233 | + omap_badwidth_write32, | |
| 2234 | + omap_badwidth_write32, | |
| 2235 | + omap_tcmi_write, | |
| 2236 | +}; | |
| 2237 | + | |
| 2238 | +static void omap_tcmi_reset(struct omap_mpu_state_s *mpu) | |
| 2239 | +{ | |
| 2240 | + mpu->tcmi_regs[0x00 >> 2] = 0x00000000; | |
| 2241 | + mpu->tcmi_regs[0x04 >> 2] = 0x00000000; | |
| 2242 | + mpu->tcmi_regs[0x08 >> 2] = 0x00000000; | |
| 2243 | + mpu->tcmi_regs[0x0c >> 2] = 0x00000010; | |
| 2244 | + mpu->tcmi_regs[0x10 >> 2] = 0x0010fffb; | |
| 2245 | + mpu->tcmi_regs[0x14 >> 2] = 0x0010fffb; | |
| 2246 | + mpu->tcmi_regs[0x18 >> 2] = 0x0010fffb; | |
| 2247 | + mpu->tcmi_regs[0x1c >> 2] = 0x0010fffb; | |
| 2248 | + mpu->tcmi_regs[0x20 >> 2] = 0x00618800; | |
| 2249 | + mpu->tcmi_regs[0x24 >> 2] = 0x00000037; | |
| 2250 | + mpu->tcmi_regs[0x28 >> 2] = 0x00000000; | |
| 2251 | + mpu->tcmi_regs[0x2c >> 2] = 0x00000000; | |
| 2252 | + mpu->tcmi_regs[0x30 >> 2] = 0x00000000; | |
| 2253 | + mpu->tcmi_regs[0x3c >> 2] = 0x00000003; | |
| 2254 | + mpu->tcmi_regs[0x40 >> 2] = 0x00000000; | |
| 2255 | +} | |
| 2256 | + | |
| 2257 | +static void omap_tcmi_init(target_phys_addr_t base, | |
| 2258 | + struct omap_mpu_state_s *mpu) | |
| 2259 | +{ | |
| 2260 | + int iomemtype = cpu_register_io_memory(0, omap_tcmi_readfn, | |
| 2261 | + omap_tcmi_writefn, mpu); | |
| 2262 | + | |
| 2263 | + mpu->tcmi_base = base; | |
| 2264 | + cpu_register_physical_memory(mpu->tcmi_base, 0x100, iomemtype); | |
| 2265 | + omap_tcmi_reset(mpu); | |
| 2266 | +} | |
| 2267 | + | |
| 2268 | +/* Digital phase-locked loops control */ | |
| 2269 | +static uint32_t omap_dpll_read(void *opaque, target_phys_addr_t addr) | |
| 2270 | +{ | |
| 2271 | + struct dpll_ctl_s *s = (struct dpll_ctl_s *) opaque; | |
| 2272 | + int offset = addr - s->base; | |
| 2273 | + | |
| 2274 | + if (offset == 0x00) /* CTL_REG */ | |
| 2275 | + return s->mode; | |
| 2276 | + | |
| 2277 | + OMAP_BAD_REG(addr); | |
| 2278 | + return 0; | |
| 2279 | +} | |
| 2280 | + | |
| 2281 | +static void omap_dpll_write(void *opaque, target_phys_addr_t addr, | |
| 2282 | + uint32_t value) | |
| 2283 | +{ | |
| 2284 | + struct dpll_ctl_s *s = (struct dpll_ctl_s *) opaque; | |
| 2285 | + uint16_t diff; | |
| 2286 | + int offset = addr - s->base; | |
| 2287 | + static const int bypass_div[4] = { 1, 2, 4, 4 }; | |
| 2288 | + int div, mult; | |
| 2289 | + | |
| 2290 | + if (offset == 0x00) { /* CTL_REG */ | |
| 2291 | + /* See omap_ulpd_pm_write() too */ | |
| 2292 | + diff = s->mode & value; | |
| 2293 | + s->mode = value & 0x2fff; | |
| 2294 | + if (diff & (0x3ff << 2)) { | |
| 2295 | + if (value & (1 << 4)) { /* PLL_ENABLE */ | |
| 2296 | + div = ((value >> 5) & 3) + 1; /* PLL_DIV */ | |
| 2297 | + mult = MIN((value >> 7) & 0x1f, 1); /* PLL_MULT */ | |
| 2298 | + } else { | |
| 2299 | + div = bypass_div[((value >> 2) & 3)]; /* BYPASS_DIV */ | |
| 2300 | + mult = 1; | |
| 2301 | + } | |
| 2302 | + omap_clk_setrate(s->dpll, div, mult); | |
| 2303 | + } | |
| 2304 | + | |
| 2305 | + /* Enter the desired mode. */ | |
| 2306 | + s->mode = (s->mode & 0xfffe) | ((s->mode >> 4) & 1); | |
| 2307 | + | |
| 2308 | + /* Act as if the lock is restored. */ | |
| 2309 | + s->mode |= 2; | |
| 2310 | + } else { | |
| 2311 | + OMAP_BAD_REG(addr); | |
| 2312 | + } | |
| 2313 | +} | |
| 2314 | + | |
| 2315 | +static CPUReadMemoryFunc *omap_dpll_readfn[] = { | |
| 2316 | + omap_badwidth_read16, | |
| 2317 | + omap_dpll_read, | |
| 2318 | + omap_badwidth_read16, | |
| 2319 | +}; | |
| 2320 | + | |
| 2321 | +static CPUWriteMemoryFunc *omap_dpll_writefn[] = { | |
| 2322 | + omap_badwidth_write16, | |
| 2323 | + omap_dpll_write, | |
| 2324 | + omap_badwidth_write16, | |
| 2325 | +}; | |
| 2326 | + | |
| 2327 | +static void omap_dpll_reset(struct dpll_ctl_s *s) | |
| 2328 | +{ | |
| 2329 | + s->mode = 0x2002; | |
| 2330 | + omap_clk_setrate(s->dpll, 1, 1); | |
| 2331 | +} | |
| 2332 | + | |
| 2333 | +static void omap_dpll_init(struct dpll_ctl_s *s, target_phys_addr_t base, | |
| 2334 | + omap_clk clk) | |
| 2335 | +{ | |
| 2336 | + int iomemtype = cpu_register_io_memory(0, omap_dpll_readfn, | |
| 2337 | + omap_dpll_writefn, s); | |
| 2338 | + | |
| 2339 | + s->base = base; | |
| 2340 | + s->dpll = clk; | |
| 2341 | + omap_dpll_reset(s); | |
| 2342 | + | |
| 2343 | + cpu_register_physical_memory(s->base, 0x100, iomemtype); | |
| 2344 | +} | |
| 2345 | + | |
| 2346 | +/* UARTs */ | |
| 2347 | +struct omap_uart_s { | |
| 2348 | + SerialState *serial; /* TODO */ | |
| 2349 | +}; | |
| 2350 | + | |
| 2351 | +static void omap_uart_reset(struct omap_uart_s *s) | |
| 2352 | +{ | |
| 2353 | +} | |
| 2354 | + | |
| 2355 | +struct omap_uart_s *omap_uart_init(target_phys_addr_t base, | |
| 2356 | + qemu_irq irq, omap_clk clk, CharDriverState *chr) | |
| 2357 | +{ | |
| 2358 | + struct omap_uart_s *s = (struct omap_uart_s *) | |
| 2359 | + qemu_mallocz(sizeof(struct omap_uart_s)); | |
| 2360 | + if (chr) | |
| 2361 | + s->serial = serial_mm_init(base, 2, irq, chr, 1); | |
| 2362 | + return s; | |
| 2363 | +} | |
| 2364 | + | |
| 2365 | +/* MPU Clock/Reset/Power Mode Control */ | |
| 2366 | +static uint32_t omap_clkm_read(void *opaque, target_phys_addr_t addr) | |
| 2367 | +{ | |
| 2368 | + struct omap_mpu_state_s *s = (struct omap_mpu_state_s *) opaque; | |
| 2369 | + int offset = addr - s->clkm.mpu_base; | |
| 2370 | + | |
| 2371 | + switch (offset) { | |
| 2372 | + case 0x00: /* ARM_CKCTL */ | |
| 2373 | + return s->clkm.arm_ckctl; | |
| 2374 | + | |
| 2375 | + case 0x04: /* ARM_IDLECT1 */ | |
| 2376 | + return s->clkm.arm_idlect1; | |
| 2377 | + | |
| 2378 | + case 0x08: /* ARM_IDLECT2 */ | |
| 2379 | + return s->clkm.arm_idlect2; | |
| 2380 | + | |
| 2381 | + case 0x0c: /* ARM_EWUPCT */ | |
| 2382 | + return s->clkm.arm_ewupct; | |
| 2383 | + | |
| 2384 | + case 0x10: /* ARM_RSTCT1 */ | |
| 2385 | + return s->clkm.arm_rstct1; | |
| 2386 | + | |
| 2387 | + case 0x14: /* ARM_RSTCT2 */ | |
| 2388 | + return s->clkm.arm_rstct2; | |
| 2389 | + | |
| 2390 | + case 0x18: /* ARM_SYSST */ | |
| 2391 | + return (s->clkm.clocking_scheme < 11) | s->clkm.cold_start; | |
| 2392 | + | |
| 2393 | + case 0x1c: /* ARM_CKOUT1 */ | |
| 2394 | + return s->clkm.arm_ckout1; | |
| 2395 | + | |
| 2396 | + case 0x20: /* ARM_CKOUT2 */ | |
| 2397 | + break; | |
| 2398 | + } | |
| 2399 | + | |
| 2400 | + OMAP_BAD_REG(addr); | |
| 2401 | + return 0; | |
| 2402 | +} | |
| 2403 | + | |
| 2404 | +static inline void omap_clkm_ckctl_update(struct omap_mpu_state_s *s, | |
| 2405 | + uint16_t diff, uint16_t value) | |
| 2406 | +{ | |
| 2407 | + omap_clk clk; | |
| 2408 | + | |
| 2409 | + if (diff & (1 << 14)) { /* ARM_INTHCK_SEL */ | |
| 2410 | + if (value & (1 << 14)) | |
| 2411 | + /* Reserved */; | |
| 2412 | + else { | |
| 2413 | + clk = omap_findclk(s, "arminth_ck"); | |
| 2414 | + omap_clk_reparent(clk, omap_findclk(s, "tc_ck")); | |
| 2415 | + } | |
| 2416 | + } | |
| 2417 | + if (diff & (1 << 12)) { /* ARM_TIMXO */ | |
| 2418 | + clk = omap_findclk(s, "armtim_ck"); | |
| 2419 | + if (value & (1 << 12)) | |
| 2420 | + omap_clk_reparent(clk, omap_findclk(s, "clkin")); | |
| 2421 | + else | |
| 2422 | + omap_clk_reparent(clk, omap_findclk(s, "ck_gen1")); | |
| 2423 | + } | |
| 2424 | + /* XXX: en_dspck */ | |
| 2425 | + if (diff & (3 << 10)) { /* DSPMMUDIV */ | |
| 2426 | + clk = omap_findclk(s, "dspmmu_ck"); | |
| 2427 | + omap_clk_setrate(clk, 1 << ((value >> 10) & 3), 1); | |
| 2428 | + } | |
| 2429 | + if (diff & (3 << 8)) { /* TCDIV */ | |
| 2430 | + clk = omap_findclk(s, "tc_ck"); | |
| 2431 | + omap_clk_setrate(clk, 1 << ((value >> 8) & 3), 1); | |
| 2432 | + } | |
| 2433 | + if (diff & (3 << 6)) { /* DSPDIV */ | |
| 2434 | + clk = omap_findclk(s, "dsp_ck"); | |
| 2435 | + omap_clk_setrate(clk, 1 << ((value >> 6) & 3), 1); | |
| 2436 | + } | |
| 2437 | + if (diff & (3 << 4)) { /* ARMDIV */ | |
| 2438 | + clk = omap_findclk(s, "arm_ck"); | |
| 2439 | + omap_clk_setrate(clk, 1 << ((value >> 4) & 3), 1); | |
| 2440 | + } | |
| 2441 | + if (diff & (3 << 2)) { /* LCDDIV */ | |
| 2442 | + clk = omap_findclk(s, "lcd_ck"); | |
| 2443 | + omap_clk_setrate(clk, 1 << ((value >> 2) & 3), 1); | |
| 2444 | + } | |
| 2445 | + if (diff & (3 << 0)) { /* PERDIV */ | |
| 2446 | + clk = omap_findclk(s, "armper_ck"); | |
| 2447 | + omap_clk_setrate(clk, 1 << ((value >> 0) & 3), 1); | |
| 2448 | + } | |
| 2449 | +} | |
| 2450 | + | |
| 2451 | +static inline void omap_clkm_idlect1_update(struct omap_mpu_state_s *s, | |
| 2452 | + uint16_t diff, uint16_t value) | |
| 2453 | +{ | |
| 2454 | + omap_clk clk; | |
| 2455 | + | |
| 2456 | + if (value & (1 << 11)) /* SETARM_IDLE */ | |
| 2457 | + cpu_interrupt(s->env, CPU_INTERRUPT_HALT); | |
| 2458 | + if (!(value & (1 << 10))) /* WKUP_MODE */ | |
| 2459 | + qemu_system_shutdown_request(); /* XXX: disable wakeup from IRQ */ | |
| 2460 | + | |
| 2461 | +#define SET_CANIDLE(clock, bit) \ | |
| 2462 | + if (diff & (1 << bit)) { \ | |
| 2463 | + clk = omap_findclk(s, clock); \ | |
| 2464 | + omap_clk_canidle(clk, (value >> bit) & 1); \ | |
| 2465 | + } | |
| 2466 | + SET_CANIDLE("mpuwd_ck", 0) /* IDLWDT_ARM */ | |
| 2467 | + SET_CANIDLE("armxor_ck", 1) /* IDLXORP_ARM */ | |
| 2468 | + SET_CANIDLE("mpuper_ck", 2) /* IDLPER_ARM */ | |
| 2469 | + SET_CANIDLE("lcd_ck", 3) /* IDLLCD_ARM */ | |
| 2470 | + SET_CANIDLE("lb_ck", 4) /* IDLLB_ARM */ | |
| 2471 | + SET_CANIDLE("hsab_ck", 5) /* IDLHSAB_ARM */ | |
| 2472 | + SET_CANIDLE("tipb_ck", 6) /* IDLIF_ARM */ | |
| 2473 | + SET_CANIDLE("dma_ck", 6) /* IDLIF_ARM */ | |
| 2474 | + SET_CANIDLE("tc_ck", 6) /* IDLIF_ARM */ | |
| 2475 | + SET_CANIDLE("dpll1", 7) /* IDLDPLL_ARM */ | |
| 2476 | + SET_CANIDLE("dpll2", 7) /* IDLDPLL_ARM */ | |
| 2477 | + SET_CANIDLE("dpll3", 7) /* IDLDPLL_ARM */ | |
| 2478 | + SET_CANIDLE("mpui_ck", 8) /* IDLAPI_ARM */ | |
| 2479 | + SET_CANIDLE("armtim_ck", 9) /* IDLTIM_ARM */ | |
| 2480 | +} | |
| 2481 | + | |
| 2482 | +static inline void omap_clkm_idlect2_update(struct omap_mpu_state_s *s, | |
| 2483 | + uint16_t diff, uint16_t value) | |
| 2484 | +{ | |
| 2485 | + omap_clk clk; | |
| 2486 | + | |
| 2487 | +#define SET_ONOFF(clock, bit) \ | |
| 2488 | + if (diff & (1 << bit)) { \ | |
| 2489 | + clk = omap_findclk(s, clock); \ | |
| 2490 | + omap_clk_onoff(clk, (value >> bit) & 1); \ | |
| 2491 | + } | |
| 2492 | + SET_ONOFF("mpuwd_ck", 0) /* EN_WDTCK */ | |
| 2493 | + SET_ONOFF("armxor_ck", 1) /* EN_XORPCK */ | |
| 2494 | + SET_ONOFF("mpuper_ck", 2) /* EN_PERCK */ | |
| 2495 | + SET_ONOFF("lcd_ck", 3) /* EN_LCDCK */ | |
| 2496 | + SET_ONOFF("lb_ck", 4) /* EN_LBCK */ | |
| 2497 | + SET_ONOFF("hsab_ck", 5) /* EN_HSABCK */ | |
| 2498 | + SET_ONOFF("mpui_ck", 6) /* EN_APICK */ | |
| 2499 | + SET_ONOFF("armtim_ck", 7) /* EN_TIMCK */ | |
| 2500 | + SET_CANIDLE("dma_ck", 8) /* DMACK_REQ */ | |
| 2501 | + SET_ONOFF("arm_gpio_ck", 9) /* EN_GPIOCK */ | |
| 2502 | + SET_ONOFF("lbfree_ck", 10) /* EN_LBFREECK */ | |
| 2503 | +} | |
| 2504 | + | |
| 2505 | +static inline void omap_clkm_ckout1_update(struct omap_mpu_state_s *s, | |
| 2506 | + uint16_t diff, uint16_t value) | |
| 2507 | +{ | |
| 2508 | + omap_clk clk; | |
| 2509 | + | |
| 2510 | + if (diff & (3 << 4)) { /* TCLKOUT */ | |
| 2511 | + clk = omap_findclk(s, "tclk_out"); | |
| 2512 | + switch ((value >> 4) & 3) { | |
| 2513 | + case 1: | |
| 2514 | + omap_clk_reparent(clk, omap_findclk(s, "ck_gen3")); | |
| 2515 | + omap_clk_onoff(clk, 1); | |
| 2516 | + break; | |
| 2517 | + case 2: | |
| 2518 | + omap_clk_reparent(clk, omap_findclk(s, "tc_ck")); | |
| 2519 | + omap_clk_onoff(clk, 1); | |
| 2520 | + break; | |
| 2521 | + default: | |
| 2522 | + omap_clk_onoff(clk, 0); | |
| 2523 | + } | |
| 2524 | + } | |
| 2525 | + if (diff & (3 << 2)) { /* DCLKOUT */ | |
| 2526 | + clk = omap_findclk(s, "dclk_out"); | |
| 2527 | + switch ((value >> 2) & 3) { | |
| 2528 | + case 0: | |
| 2529 | + omap_clk_reparent(clk, omap_findclk(s, "dspmmu_ck")); | |
| 2530 | + break; | |
| 2531 | + case 1: | |
| 2532 | + omap_clk_reparent(clk, omap_findclk(s, "ck_gen2")); | |
| 2533 | + break; | |
| 2534 | + case 2: | |
| 2535 | + omap_clk_reparent(clk, omap_findclk(s, "dsp_ck")); | |
| 2536 | + break; | |
| 2537 | + case 3: | |
| 2538 | + omap_clk_reparent(clk, omap_findclk(s, "ck_ref14")); | |
| 2539 | + break; | |
| 2540 | + } | |
| 2541 | + } | |
| 2542 | + if (diff & (3 << 0)) { /* ACLKOUT */ | |
| 2543 | + clk = omap_findclk(s, "aclk_out"); | |
| 2544 | + switch ((value >> 0) & 3) { | |
| 2545 | + case 1: | |
| 2546 | + omap_clk_reparent(clk, omap_findclk(s, "ck_gen1")); | |
| 2547 | + omap_clk_onoff(clk, 1); | |
| 2548 | + break; | |
| 2549 | + case 2: | |
| 2550 | + omap_clk_reparent(clk, omap_findclk(s, "arm_ck")); | |
| 2551 | + omap_clk_onoff(clk, 1); | |
| 2552 | + break; | |
| 2553 | + case 3: | |
| 2554 | + omap_clk_reparent(clk, omap_findclk(s, "ck_ref14")); | |
| 2555 | + omap_clk_onoff(clk, 1); | |
| 2556 | + break; | |
| 2557 | + default: | |
| 2558 | + omap_clk_onoff(clk, 0); | |
| 2559 | + } | |
| 2560 | + } | |
| 2561 | +} | |
| 2562 | + | |
| 2563 | +static void omap_clkm_write(void *opaque, target_phys_addr_t addr, | |
| 2564 | + uint32_t value) | |
| 2565 | +{ | |
| 2566 | + struct omap_mpu_state_s *s = (struct omap_mpu_state_s *) opaque; | |
| 2567 | + int offset = addr - s->clkm.mpu_base; | |
| 2568 | + uint16_t diff; | |
| 2569 | + omap_clk clk; | |
| 2570 | + static const char *clkschemename[8] = { | |
| 2571 | + "fully synchronous", "fully asynchronous", "synchronous scalable", | |
| 2572 | + "mix mode 1", "mix mode 2", "bypass mode", "mix mode 3", "mix mode 4", | |
| 2573 | + }; | |
| 2574 | + | |
| 2575 | + switch (offset) { | |
| 2576 | + case 0x00: /* ARM_CKCTL */ | |
| 2577 | + diff = s->clkm.arm_ckctl ^ value; | |
| 2578 | + s->clkm.arm_ckctl = value & 0x7fff; | |
| 2579 | + omap_clkm_ckctl_update(s, diff, value); | |
| 2580 | + return; | |
| 2581 | + | |
| 2582 | + case 0x04: /* ARM_IDLECT1 */ | |
| 2583 | + diff = s->clkm.arm_idlect1 ^ value; | |
| 2584 | + s->clkm.arm_idlect1 = value & 0x0fff; | |
| 2585 | + omap_clkm_idlect1_update(s, diff, value); | |
| 2586 | + return; | |
| 2587 | + | |
| 2588 | + case 0x08: /* ARM_IDLECT2 */ | |
| 2589 | + diff = s->clkm.arm_idlect2 ^ value; | |
| 2590 | + s->clkm.arm_idlect2 = value & 0x07ff; | |
| 2591 | + omap_clkm_idlect2_update(s, diff, value); | |
| 2592 | + return; | |
| 2593 | + | |
| 2594 | + case 0x0c: /* ARM_EWUPCT */ | |
| 2595 | + diff = s->clkm.arm_ewupct ^ value; | |
| 2596 | + s->clkm.arm_ewupct = value & 0x003f; | |
| 2597 | + return; | |
| 2598 | + | |
| 2599 | + case 0x10: /* ARM_RSTCT1 */ | |
| 2600 | + diff = s->clkm.arm_rstct1 ^ value; | |
| 2601 | + s->clkm.arm_rstct1 = value & 0x0007; | |
| 2602 | + if (value & 9) { | |
| 2603 | + qemu_system_reset_request(); | |
| 2604 | + s->clkm.cold_start = 0xa; | |
| 2605 | + } | |
| 2606 | + if (diff & ~value & 4) { /* DSP_RST */ | |
| 2607 | + omap_mpui_reset(s); | |
| 2608 | + omap_tipb_bridge_reset(s->private_tipb); | |
| 2609 | + omap_tipb_bridge_reset(s->public_tipb); | |
| 2610 | + } | |
| 2611 | + if (diff & 2) { /* DSP_EN */ | |
| 2612 | + clk = omap_findclk(s, "dsp_ck"); | |
| 2613 | + omap_clk_canidle(clk, (~value >> 1) & 1); | |
| 2614 | + } | |
| 2615 | + return; | |
| 2616 | + | |
| 2617 | + case 0x14: /* ARM_RSTCT2 */ | |
| 2618 | + s->clkm.arm_rstct2 = value & 0x0001; | |
| 2619 | + return; | |
| 2620 | + | |
| 2621 | + case 0x18: /* ARM_SYSST */ | |
| 2622 | + if ((s->clkm.clocking_scheme ^ (value >> 11)) & 7) { | |
| 2623 | + s->clkm.clocking_scheme = (value >> 11) & 7; | |
| 2624 | + printf("%s: clocking scheme set to %s\n", __FUNCTION__, | |
| 2625 | + clkschemename[s->clkm.clocking_scheme]); | |
| 2626 | + } | |
| 2627 | + s->clkm.cold_start &= value & 0x3f; | |
| 2628 | + return; | |
| 2629 | + | |
| 2630 | + case 0x1c: /* ARM_CKOUT1 */ | |
| 2631 | + diff = s->clkm.arm_ckout1 ^ value; | |
| 2632 | + s->clkm.arm_ckout1 = value & 0x003f; | |
| 2633 | + omap_clkm_ckout1_update(s, diff, value); | |
| 2634 | + return; | |
| 2635 | + | |
| 2636 | + case 0x20: /* ARM_CKOUT2 */ | |
| 2637 | + default: | |
| 2638 | + OMAP_BAD_REG(addr); | |
| 2639 | + } | |
| 2640 | +} | |
| 2641 | + | |
| 2642 | +static CPUReadMemoryFunc *omap_clkm_readfn[] = { | |
| 2643 | + omap_badwidth_read16, | |
| 2644 | + omap_clkm_read, | |
| 2645 | + omap_badwidth_read16, | |
| 2646 | +}; | |
| 2647 | + | |
| 2648 | +static CPUWriteMemoryFunc *omap_clkm_writefn[] = { | |
| 2649 | + omap_badwidth_write16, | |
| 2650 | + omap_clkm_write, | |
| 2651 | + omap_badwidth_write16, | |
| 2652 | +}; | |
| 2653 | + | |
| 2654 | +static uint32_t omap_clkdsp_read(void *opaque, target_phys_addr_t addr) | |
| 2655 | +{ | |
| 2656 | + struct omap_mpu_state_s *s = (struct omap_mpu_state_s *) opaque; | |
| 2657 | + int offset = addr - s->clkm.dsp_base; | |
| 2658 | + | |
| 2659 | + switch (offset) { | |
| 2660 | + case 0x04: /* DSP_IDLECT1 */ | |
| 2661 | + return s->clkm.dsp_idlect1; | |
| 2662 | + | |
| 2663 | + case 0x08: /* DSP_IDLECT2 */ | |
| 2664 | + return s->clkm.dsp_idlect2; | |
| 2665 | + | |
| 2666 | + case 0x14: /* DSP_RSTCT2 */ | |
| 2667 | + return s->clkm.dsp_rstct2; | |
| 2668 | + | |
| 2669 | + case 0x18: /* DSP_SYSST */ | |
| 2670 | + return (s->clkm.clocking_scheme < 11) | s->clkm.cold_start | | |
| 2671 | + (s->env->halted << 6); /* Quite useless... */ | |
| 2672 | + } | |
| 2673 | + | |
| 2674 | + OMAP_BAD_REG(addr); | |
| 2675 | + return 0; | |
| 2676 | +} | |
| 2677 | + | |
| 2678 | +static inline void omap_clkdsp_idlect1_update(struct omap_mpu_state_s *s, | |
| 2679 | + uint16_t diff, uint16_t value) | |
| 2680 | +{ | |
| 2681 | + omap_clk clk; | |
| 2682 | + | |
| 2683 | + SET_CANIDLE("dspxor_ck", 1); /* IDLXORP_DSP */ | |
| 2684 | +} | |
| 2685 | + | |
| 2686 | +static inline void omap_clkdsp_idlect2_update(struct omap_mpu_state_s *s, | |
| 2687 | + uint16_t diff, uint16_t value) | |
| 2688 | +{ | |
| 2689 | + omap_clk clk; | |
| 2690 | + | |
| 2691 | + SET_ONOFF("dspxor_ck", 1); /* EN_XORPCK */ | |
| 2692 | +} | |
| 2693 | + | |
| 2694 | +static void omap_clkdsp_write(void *opaque, target_phys_addr_t addr, | |
| 2695 | + uint32_t value) | |
| 2696 | +{ | |
| 2697 | + struct omap_mpu_state_s *s = (struct omap_mpu_state_s *) opaque; | |
| 2698 | + int offset = addr - s->clkm.dsp_base; | |
| 2699 | + uint16_t diff; | |
| 2700 | + | |
| 2701 | + switch (offset) { | |
| 2702 | + case 0x04: /* DSP_IDLECT1 */ | |
| 2703 | + diff = s->clkm.dsp_idlect1 ^ value; | |
| 2704 | + s->clkm.dsp_idlect1 = value & 0x01f7; | |
| 2705 | + omap_clkdsp_idlect1_update(s, diff, value); | |
| 2706 | + break; | |
| 2707 | + | |
| 2708 | + case 0x08: /* DSP_IDLECT2 */ | |
| 2709 | + s->clkm.dsp_idlect2 = value & 0x0037; | |
| 2710 | + diff = s->clkm.dsp_idlect1 ^ value; | |
| 2711 | + omap_clkdsp_idlect2_update(s, diff, value); | |
| 2712 | + break; | |
| 2713 | + | |
| 2714 | + case 0x14: /* DSP_RSTCT2 */ | |
| 2715 | + s->clkm.dsp_rstct2 = value & 0x0001; | |
| 2716 | + break; | |
| 2717 | + | |
| 2718 | + case 0x18: /* DSP_SYSST */ | |
| 2719 | + s->clkm.cold_start &= value & 0x3f; | |
| 2720 | + break; | |
| 2721 | + | |
| 2722 | + default: | |
| 2723 | + OMAP_BAD_REG(addr); | |
| 2724 | + } | |
| 2725 | +} | |
| 2726 | + | |
| 2727 | +static CPUReadMemoryFunc *omap_clkdsp_readfn[] = { | |
| 2728 | + omap_badwidth_read16, | |
| 2729 | + omap_clkdsp_read, | |
| 2730 | + omap_badwidth_read16, | |
| 2731 | +}; | |
| 2732 | + | |
| 2733 | +static CPUWriteMemoryFunc *omap_clkdsp_writefn[] = { | |
| 2734 | + omap_badwidth_write16, | |
| 2735 | + omap_clkdsp_write, | |
| 2736 | + omap_badwidth_write16, | |
| 2737 | +}; | |
| 2738 | + | |
| 2739 | +static void omap_clkm_reset(struct omap_mpu_state_s *s) | |
| 2740 | +{ | |
| 2741 | + if (s->wdt && s->wdt->reset) | |
| 2742 | + s->clkm.cold_start = 0x6; | |
| 2743 | + s->clkm.clocking_scheme = 0; | |
| 2744 | + omap_clkm_ckctl_update(s, ~0, 0x3000); | |
| 2745 | + s->clkm.arm_ckctl = 0x3000; | |
| 2746 | + omap_clkm_idlect1_update(s, s->clkm.arm_idlect1 & 0x0400, 0x0400); | |
| 2747 | + s->clkm.arm_idlect1 = 0x0400; | |
| 2748 | + omap_clkm_idlect2_update(s, s->clkm.arm_idlect2 & 0x0100, 0x0100); | |
| 2749 | + s->clkm.arm_idlect2 = 0x0100; | |
| 2750 | + s->clkm.arm_ewupct = 0x003f; | |
| 2751 | + s->clkm.arm_rstct1 = 0x0000; | |
| 2752 | + s->clkm.arm_rstct2 = 0x0000; | |
| 2753 | + s->clkm.arm_ckout1 = 0x0015; | |
| 2754 | + s->clkm.dpll1_mode = 0x2002; | |
| 2755 | + omap_clkdsp_idlect1_update(s, s->clkm.dsp_idlect1 ^ 0x0040, 0x0040); | |
| 2756 | + s->clkm.dsp_idlect1 = 0x0040; | |
| 2757 | + omap_clkdsp_idlect2_update(s, ~0, 0x0000); | |
| 2758 | + s->clkm.dsp_idlect2 = 0x0000; | |
| 2759 | + s->clkm.dsp_rstct2 = 0x0000; | |
| 2760 | +} | |
| 2761 | + | |
| 2762 | +static void omap_clkm_init(target_phys_addr_t mpu_base, | |
| 2763 | + target_phys_addr_t dsp_base, struct omap_mpu_state_s *s) | |
| 2764 | +{ | |
| 2765 | + int iomemtype[2] = { | |
| 2766 | + cpu_register_io_memory(0, omap_clkm_readfn, omap_clkm_writefn, s), | |
| 2767 | + cpu_register_io_memory(0, omap_clkdsp_readfn, omap_clkdsp_writefn, s), | |
| 2768 | + }; | |
| 2769 | + | |
| 2770 | + s->clkm.mpu_base = mpu_base; | |
| 2771 | + s->clkm.dsp_base = dsp_base; | |
| 2772 | + s->clkm.cold_start = 0x3a; | |
| 2773 | + omap_clkm_reset(s); | |
| 2774 | + | |
| 2775 | + cpu_register_physical_memory(s->clkm.mpu_base, 0x100, iomemtype[0]); | |
| 2776 | + cpu_register_physical_memory(s->clkm.dsp_base, 0x1000, iomemtype[1]); | |
| 2777 | +} | |
| 2778 | + | |
| 2779 | +/* General chip reset */ | |
| 2780 | +static void omap_mpu_reset(void *opaque) | |
| 2781 | +{ | |
| 2782 | + struct omap_mpu_state_s *mpu = (struct omap_mpu_state_s *) opaque; | |
| 2783 | + | |
| 2784 | + omap_clkm_reset(mpu); | |
| 2785 | + omap_inth_reset(mpu->ih[0]); | |
| 2786 | + omap_inth_reset(mpu->ih[1]); | |
| 2787 | + omap_dma_reset(mpu->dma); | |
| 2788 | + omap_mpu_timer_reset(mpu->timer[0]); | |
| 2789 | + omap_mpu_timer_reset(mpu->timer[1]); | |
| 2790 | + omap_mpu_timer_reset(mpu->timer[2]); | |
| 2791 | + omap_wd_timer_reset(mpu->wdt); | |
| 2792 | + omap_os_timer_reset(mpu->os_timer); | |
| 2793 | + omap_lcdc_reset(mpu->lcd); | |
| 2794 | + omap_ulpd_pm_reset(mpu); | |
| 2795 | + omap_pin_cfg_reset(mpu); | |
| 2796 | + omap_mpui_reset(mpu); | |
| 2797 | + omap_tipb_bridge_reset(mpu->private_tipb); | |
| 2798 | + omap_tipb_bridge_reset(mpu->public_tipb); | |
| 2799 | + omap_dpll_reset(&mpu->dpll[0]); | |
| 2800 | + omap_dpll_reset(&mpu->dpll[1]); | |
| 2801 | + omap_dpll_reset(&mpu->dpll[2]); | |
| 2802 | + omap_uart_reset(mpu->uart1); | |
| 2803 | + omap_uart_reset(mpu->uart2); | |
| 2804 | + omap_uart_reset(mpu->uart3); | |
| 2805 | + cpu_reset(mpu->env); | |
| 2806 | +} | |
| 2807 | + | |
| 2808 | +static void omap_mpu_wakeup(void *opaque, int irq, int req) | |
| 2809 | +{ | |
| 2810 | + struct omap_mpu_state_s *mpu = (struct omap_mpu_state_s *) opaque; | |
| 2811 | + | |
| 2812 | + cpu_interrupt(mpu->env, CPU_INTERRUPT_EXITTB); | |
| 2813 | +} | |
| 2814 | + | |
| 2815 | +struct omap_mpu_state_s *omap310_mpu_init(unsigned long sdram_size, | |
| 2816 | + DisplayState *ds, const char *core) | |
| 2817 | +{ | |
| 2818 | + struct omap_mpu_state_s *s = (struct omap_mpu_state_s *) | |
| 2819 | + qemu_mallocz(sizeof(struct omap_mpu_state_s)); | |
| 2820 | + ram_addr_t imif_base, emiff_base; | |
| 2821 | + | |
| 2822 | + /* Core */ | |
| 2823 | + s->mpu_model = omap310; | |
| 2824 | + s->env = cpu_init(); | |
| 2825 | + s->sdram_size = sdram_size; | |
| 2826 | + s->sram_size = OMAP15XX_SRAM_SIZE; | |
| 2827 | + | |
| 2828 | + cpu_arm_set_model(s->env, core ?: "ti925t"); | |
| 2829 | + | |
| 2830 | + /* Clocks */ | |
| 2831 | + omap_clk_init(s); | |
| 2832 | + | |
| 2833 | + /* Memory-mapped stuff */ | |
| 2834 | + cpu_register_physical_memory(OMAP_EMIFF_BASE, s->sdram_size, | |
| 2835 | + (emiff_base = qemu_ram_alloc(s->sdram_size)) | IO_MEM_RAM); | |
| 2836 | + cpu_register_physical_memory(OMAP_IMIF_BASE, s->sram_size, | |
| 2837 | + (imif_base = qemu_ram_alloc(s->sram_size)) | IO_MEM_RAM); | |
| 2838 | + | |
| 2839 | + omap_clkm_init(0xfffece00, 0xe1008000, s); | |
| 2840 | + | |
| 2841 | + s->ih[0] = omap_inth_init(0xfffecb00, 0x100, | |
| 2842 | + arm_pic_init_cpu(s->env), | |
| 2843 | + omap_findclk(s, "arminth_ck")); | |
| 2844 | + s->ih[1] = omap_inth_init(0xfffe0000, 0x800, | |
| 2845 | + &s->ih[0]->pins[OMAP_INT_15XX_IH2_IRQ], | |
| 2846 | + omap_findclk(s, "arminth_ck")); | |
| 2847 | + s->irq[0] = s->ih[0]->pins; | |
| 2848 | + s->irq[1] = s->ih[1]->pins; | |
| 2849 | + | |
| 2850 | + s->dma = omap_dma_init(0xfffed800, s->irq[0], s, | |
| 2851 | + omap_findclk(s, "dma_ck")); | |
| 2852 | + s->port[emiff ].addr_valid = omap_validate_emiff_addr; | |
| 2853 | + s->port[emifs ].addr_valid = omap_validate_emifs_addr; | |
| 2854 | + s->port[imif ].addr_valid = omap_validate_imif_addr; | |
| 2855 | + s->port[tipb ].addr_valid = omap_validate_tipb_addr; | |
| 2856 | + s->port[local ].addr_valid = omap_validate_local_addr; | |
| 2857 | + s->port[tipb_mpui].addr_valid = omap_validate_tipb_mpui_addr; | |
| 2858 | + | |
| 2859 | + s->timer[0] = omap_mpu_timer_init(0xfffec500, | |
| 2860 | + s->irq[0][OMAP_INT_TIMER1], | |
| 2861 | + omap_findclk(s, "mputim_ck")); | |
| 2862 | + s->timer[1] = omap_mpu_timer_init(0xfffec600, | |
| 2863 | + s->irq[0][OMAP_INT_TIMER2], | |
| 2864 | + omap_findclk(s, "mputim_ck")); | |
| 2865 | + s->timer[2] = omap_mpu_timer_init(0xfffec700, | |
| 2866 | + s->irq[0][OMAP_INT_TIMER3], | |
| 2867 | + omap_findclk(s, "mputim_ck")); | |
| 2868 | + | |
| 2869 | + s->wdt = omap_wd_timer_init(0xfffec800, | |
| 2870 | + s->irq[0][OMAP_INT_WD_TIMER], | |
| 2871 | + omap_findclk(s, "armwdt_ck")); | |
| 2872 | + | |
| 2873 | + s->os_timer = omap_os_timer_init(0xfffb9000, | |
| 2874 | + s->irq[1][OMAP_INT_OS_TIMER], | |
| 2875 | + omap_findclk(s, "clk32-kHz")); | |
| 2876 | + | |
| 2877 | + s->lcd = omap_lcdc_init(0xfffec000, s->irq[0][OMAP_INT_LCD_CTRL], | |
| 2878 | + &s->dma->lcd_ch, ds, imif_base, emiff_base, | |
| 2879 | + omap_findclk(s, "lcd_ck")); | |
| 2880 | + | |
| 2881 | + omap_ulpd_pm_init(0xfffe0800, s); | |
| 2882 | + omap_pin_cfg_init(0xfffe1000, s); | |
| 2883 | + omap_id_init(s); | |
| 2884 | + | |
| 2885 | + omap_mpui_init(0xfffec900, s); | |
| 2886 | + | |
| 2887 | + s->private_tipb = omap_tipb_bridge_init(0xfffeca00, | |
| 2888 | + s->irq[0][OMAP_INT_BRIDGE_PRIV], | |
| 2889 | + omap_findclk(s, "tipb_ck")); | |
| 2890 | + s->public_tipb = omap_tipb_bridge_init(0xfffed300, | |
| 2891 | + s->irq[0][OMAP_INT_BRIDGE_PUB], | |
| 2892 | + omap_findclk(s, "tipb_ck")); | |
| 2893 | + | |
| 2894 | + omap_tcmi_init(0xfffecc00, s); | |
| 2895 | + | |
| 2896 | + s->uart1 = omap_uart_init(0xfffb0000, s->irq[1][OMAP_INT_UART1], | |
| 2897 | + omap_findclk(s, "uart1_ck"), | |
| 2898 | + serial_hds[0]); | |
| 2899 | + s->uart2 = omap_uart_init(0xfffb0800, s->irq[1][OMAP_INT_UART2], | |
| 2900 | + omap_findclk(s, "uart2_ck"), | |
| 2901 | + serial_hds[0] ? serial_hds[1] : 0); | |
| 2902 | + s->uart3 = omap_uart_init(0xe1019800, s->irq[0][OMAP_INT_UART3], | |
| 2903 | + omap_findclk(s, "uart3_ck"), | |
| 2904 | + serial_hds[0] && serial_hds[1] ? serial_hds[2] : 0); | |
| 2905 | + | |
| 2906 | + omap_dpll_init(&s->dpll[0], 0xfffecf00, omap_findclk(s, "dpll1")); | |
| 2907 | + omap_dpll_init(&s->dpll[1], 0xfffed000, omap_findclk(s, "dpll2")); | |
| 2908 | + omap_dpll_init(&s->dpll[2], 0xfffed100, omap_findclk(s, "dpll3")); | |
| 2909 | + | |
| 2910 | + qemu_register_reset(omap_mpu_reset, s); | |
| 2911 | + s->wakeup = qemu_allocate_irqs(omap_mpu_wakeup, s, 1)[0]; | |
| 2912 | + | |
| 2913 | + return s; | |
| 2914 | +} | ... | ... |
hw/omap.h
0 → 100644
| 1 | +/* | |
| 2 | + * Texas Instruments OMAP processors. | |
| 3 | + * | |
| 4 | + * Copyright (C) 2006-2007 Andrzej Zaborowski <balrog@zabor.org> | |
| 5 | + * | |
| 6 | + * This program is free software; you can redistribute it and/or | |
| 7 | + * modify it under the terms of the GNU General Public License as | |
| 8 | + * published by the Free Software Foundation; either version 2 of | |
| 9 | + * the License, or (at your option) any later version. | |
| 10 | + * | |
| 11 | + * This program is distributed in the hope that it will be useful, | |
| 12 | + * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
| 13 | + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
| 14 | + * GNU General Public License for more details. | |
| 15 | + * | |
| 16 | + * You should have received a copy of the GNU General Public License | |
| 17 | + * along with this program; if not, write to the Free Software | |
| 18 | + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, | |
| 19 | + * MA 02111-1307 USA | |
| 20 | + */ | |
| 21 | +#ifndef hw_omap_h | |
| 22 | +# define hw_omap_h "omap.h" | |
| 23 | + | |
| 24 | +# define OMAP_EMIFS_BASE 0x00000000 | |
| 25 | +# define OMAP_CS0_BASE 0x00000000 | |
| 26 | +# define OMAP_CS1_BASE 0x04000000 | |
| 27 | +# define OMAP_CS2_BASE 0x08000000 | |
| 28 | +# define OMAP_CS3_BASE 0x0c000000 | |
| 29 | +# define OMAP_EMIFF_BASE 0x10000000 | |
| 30 | +# define OMAP_IMIF_BASE 0x20000000 | |
| 31 | +# define OMAP_LOCALBUS_BASE 0x30000000 | |
| 32 | +# define OMAP_MPUI_BASE 0xe1000000 | |
| 33 | + | |
| 34 | +# define OMAP730_SRAM_SIZE 0x00032000 | |
| 35 | +# define OMAP15XX_SRAM_SIZE 0x00030000 | |
| 36 | +# define OMAP16XX_SRAM_SIZE 0x00004000 | |
| 37 | +# define OMAP1611_SRAM_SIZE 0x0003e800 | |
| 38 | +# define OMAP_CS0_SIZE 0x04000000 | |
| 39 | +# define OMAP_CS1_SIZE 0x04000000 | |
| 40 | +# define OMAP_CS2_SIZE 0x04000000 | |
| 41 | +# define OMAP_CS3_SIZE 0x04000000 | |
| 42 | + | |
| 43 | +/* omap1_clk.c */ | |
| 44 | +struct omap_mpu_state_s; | |
| 45 | +typedef struct clk *omap_clk; | |
| 46 | +omap_clk omap_findclk(struct omap_mpu_state_s *mpu, const char *name); | |
| 47 | +void omap_clk_init(struct omap_mpu_state_s *mpu); | |
| 48 | +void omap_clk_adduser(struct clk *clk, qemu_irq user); | |
| 49 | +void omap_clk_get(omap_clk clk); | |
| 50 | +void omap_clk_put(omap_clk clk); | |
| 51 | +void omap_clk_onoff(omap_clk clk, int on); | |
| 52 | +void omap_clk_canidle(omap_clk clk, int can); | |
| 53 | +void omap_clk_setrate(omap_clk clk, int divide, int multiply); | |
| 54 | +int64_t omap_clk_getrate(omap_clk clk); | |
| 55 | +void omap_clk_reparent(omap_clk clk, omap_clk parent); | |
| 56 | + | |
| 57 | +/* omap.c */ | |
| 58 | +struct omap_intr_handler_s; | |
| 59 | +struct omap_intr_handler_s *omap_inth_init(target_phys_addr_t base, | |
| 60 | + unsigned long size, qemu_irq parent[2], omap_clk clk); | |
| 61 | + | |
| 62 | +/* | |
| 63 | + * Common IRQ numbers for level 1 interrupt handler | |
| 64 | + * See /usr/include/asm-arm/arch-omap/irqs.h in Linux. | |
| 65 | + */ | |
| 66 | +# define OMAP_INT_CAMERA 1 | |
| 67 | +# define OMAP_INT_FIQ 3 | |
| 68 | +# define OMAP_INT_RTDX 6 | |
| 69 | +# define OMAP_INT_DSP_MMU_ABORT 7 | |
| 70 | +# define OMAP_INT_HOST 8 | |
| 71 | +# define OMAP_INT_ABORT 9 | |
| 72 | +# define OMAP_INT_BRIDGE_PRIV 13 | |
| 73 | +# define OMAP_INT_GPIO_BANK1 14 | |
| 74 | +# define OMAP_INT_UART3 15 | |
| 75 | +# define OMAP_INT_TIMER3 16 | |
| 76 | +# define OMAP_INT_DMA_CH0_6 19 | |
| 77 | +# define OMAP_INT_DMA_CH1_7 20 | |
| 78 | +# define OMAP_INT_DMA_CH2_8 21 | |
| 79 | +# define OMAP_INT_DMA_CH3 22 | |
| 80 | +# define OMAP_INT_DMA_CH4 23 | |
| 81 | +# define OMAP_INT_DMA_CH5 24 | |
| 82 | +# define OMAP_INT_DMA_LCD 25 | |
| 83 | +# define OMAP_INT_TIMER1 26 | |
| 84 | +# define OMAP_INT_WD_TIMER 27 | |
| 85 | +# define OMAP_INT_BRIDGE_PUB 28 | |
| 86 | +# define OMAP_INT_TIMER2 30 | |
| 87 | +# define OMAP_INT_LCD_CTRL 31 | |
| 88 | + | |
| 89 | +/* | |
| 90 | + * Common OMAP-15xx IRQ numbers for level 1 interrupt handler | |
| 91 | + */ | |
| 92 | +# define OMAP_INT_15XX_IH2_IRQ 0 | |
| 93 | +# define OMAP_INT_15XX_LB_MMU 17 | |
| 94 | +# define OMAP_INT_15XX_LOCAL_BUS 29 | |
| 95 | + | |
| 96 | +/* | |
| 97 | + * OMAP-1510 specific IRQ numbers for level 1 interrupt handler | |
| 98 | + */ | |
| 99 | +# define OMAP_INT_1510_SPI_TX 4 | |
| 100 | +# define OMAP_INT_1510_SPI_RX 5 | |
| 101 | +# define OMAP_INT_1510_DSP_MAILBOX1 10 | |
| 102 | +# define OMAP_INT_1510_DSP_MAILBOX2 11 | |
| 103 | + | |
| 104 | +/* | |
| 105 | + * OMAP-310 specific IRQ numbers for level 1 interrupt handler | |
| 106 | + */ | |
| 107 | +# define OMAP_INT_310_McBSP2_TX 4 | |
| 108 | +# define OMAP_INT_310_McBSP2_RX 5 | |
| 109 | +# define OMAP_INT_310_HSB_MAILBOX1 12 | |
| 110 | +# define OMAP_INT_310_HSAB_MMU 18 | |
| 111 | + | |
| 112 | +/* | |
| 113 | + * OMAP-1610 specific IRQ numbers for level 1 interrupt handler | |
| 114 | + */ | |
| 115 | +# define OMAP_INT_1610_IH2_IRQ 0 | |
| 116 | +# define OMAP_INT_1610_IH2_FIQ 2 | |
| 117 | +# define OMAP_INT_1610_McBSP2_TX 4 | |
| 118 | +# define OMAP_INT_1610_McBSP2_RX 5 | |
| 119 | +# define OMAP_INT_1610_DSP_MAILBOX1 10 | |
| 120 | +# define OMAP_INT_1610_DSP_MAILBOX2 11 | |
| 121 | +# define OMAP_INT_1610_LCD_LINE 12 | |
| 122 | +# define OMAP_INT_1610_GPTIMER1 17 | |
| 123 | +# define OMAP_INT_1610_GPTIMER2 18 | |
| 124 | +# define OMAP_INT_1610_SSR_FIFO_0 29 | |
| 125 | + | |
| 126 | +/* | |
| 127 | + * OMAP-730 specific IRQ numbers for level 1 interrupt handler | |
| 128 | + */ | |
| 129 | +# define OMAP_INT_730_IH2_FIQ 0 | |
| 130 | +# define OMAP_INT_730_IH2_IRQ 1 | |
| 131 | +# define OMAP_INT_730_USB_NON_ISO 2 | |
| 132 | +# define OMAP_INT_730_USB_ISO 3 | |
| 133 | +# define OMAP_INT_730_ICR 4 | |
| 134 | +# define OMAP_INT_730_EAC 5 | |
| 135 | +# define OMAP_INT_730_GPIO_BANK1 6 | |
| 136 | +# define OMAP_INT_730_GPIO_BANK2 7 | |
| 137 | +# define OMAP_INT_730_GPIO_BANK3 8 | |
| 138 | +# define OMAP_INT_730_McBSP2TX 10 | |
| 139 | +# define OMAP_INT_730_McBSP2RX 11 | |
| 140 | +# define OMAP_INT_730_McBSP2RX_OVF 12 | |
| 141 | +# define OMAP_INT_730_LCD_LINE 14 | |
| 142 | +# define OMAP_INT_730_GSM_PROTECT 15 | |
| 143 | +# define OMAP_INT_730_TIMER3 16 | |
| 144 | +# define OMAP_INT_730_GPIO_BANK5 17 | |
| 145 | +# define OMAP_INT_730_GPIO_BANK6 18 | |
| 146 | +# define OMAP_INT_730_SPGIO_WR 29 | |
| 147 | + | |
| 148 | +/* | |
| 149 | + * Common IRQ numbers for level 2 interrupt handler | |
| 150 | + */ | |
| 151 | +# define OMAP_INT_KEYBOARD 1 | |
| 152 | +# define OMAP_INT_uWireTX 2 | |
| 153 | +# define OMAP_INT_uWireRX 3 | |
| 154 | +# define OMAP_INT_I2C 4 | |
| 155 | +# define OMAP_INT_MPUIO 5 | |
| 156 | +# define OMAP_INT_USB_HHC_1 6 | |
| 157 | +# define OMAP_INT_McBSP3TX 10 | |
| 158 | +# define OMAP_INT_McBSP3RX 11 | |
| 159 | +# define OMAP_INT_McBSP1TX 12 | |
| 160 | +# define OMAP_INT_McBSP1RX 13 | |
| 161 | +# define OMAP_INT_UART1 14 | |
| 162 | +# define OMAP_INT_UART2 15 | |
| 163 | +# define OMAP_INT_USB_W2FC 20 | |
| 164 | +# define OMAP_INT_1WIRE 21 | |
| 165 | +# define OMAP_INT_OS_TIMER 22 | |
| 166 | +# define OMAP_INT_MMC 23 | |
| 167 | +# define OMAP_INT_GAUGE_32K 24 | |
| 168 | +# define OMAP_INT_RTC_TIMER 25 | |
| 169 | +# define OMAP_INT_RTC_ALARM 26 | |
| 170 | +# define OMAP_INT_DSP_MMU 28 | |
| 171 | + | |
| 172 | +/* | |
| 173 | + * OMAP-1510 specific IRQ numbers for level 2 interrupt handler | |
| 174 | + */ | |
| 175 | +# define OMAP_INT_1510_BT_MCSI1TX 16 | |
| 176 | +# define OMAP_INT_1510_BT_MCSI1RX 17 | |
| 177 | +# define OMAP_INT_1510_SoSSI_MATCH 19 | |
| 178 | +# define OMAP_INT_1510_MEM_STICK 27 | |
| 179 | +# define OMAP_INT_1510_COM_SPI_RO 31 | |
| 180 | + | |
| 181 | +/* | |
| 182 | + * OMAP-310 specific IRQ numbers for level 2 interrupt handler | |
| 183 | + */ | |
| 184 | +# define OMAP_INT_310_FAC 0 | |
| 185 | +# define OMAP_INT_310_USB_HHC_2 7 | |
| 186 | +# define OMAP_INT_310_MCSI1_FE 16 | |
| 187 | +# define OMAP_INT_310_MCSI2_FE 17 | |
| 188 | +# define OMAP_INT_310_USB_W2FC_ISO 29 | |
| 189 | +# define OMAP_INT_310_USB_W2FC_NON_ISO 30 | |
| 190 | +# define OMAP_INT_310_McBSP2RX_OF 31 | |
| 191 | + | |
| 192 | +/* | |
| 193 | + * OMAP-1610 specific IRQ numbers for level 2 interrupt handler | |
| 194 | + */ | |
| 195 | +# define OMAP_INT_1610_FAC 0 | |
| 196 | +# define OMAP_INT_1610_USB_HHC_2 7 | |
| 197 | +# define OMAP_INT_1610_USB_OTG 8 | |
| 198 | +# define OMAP_INT_1610_SoSSI 9 | |
| 199 | +# define OMAP_INT_1610_BT_MCSI1TX 16 | |
| 200 | +# define OMAP_INT_1610_BT_MCSI1RX 17 | |
| 201 | +# define OMAP_INT_1610_SoSSI_MATCH 19 | |
| 202 | +# define OMAP_INT_1610_MEM_STICK 27 | |
| 203 | +# define OMAP_INT_1610_McBSP2RX_OF 31 | |
| 204 | +# define OMAP_INT_1610_STI 32 | |
| 205 | +# define OMAP_INT_1610_STI_WAKEUP 33 | |
| 206 | +# define OMAP_INT_1610_GPTIMER3 34 | |
| 207 | +# define OMAP_INT_1610_GPTIMER4 35 | |
| 208 | +# define OMAP_INT_1610_GPTIMER5 36 | |
| 209 | +# define OMAP_INT_1610_GPTIMER6 37 | |
| 210 | +# define OMAP_INT_1610_GPTIMER7 38 | |
| 211 | +# define OMAP_INT_1610_GPTIMER8 39 | |
| 212 | +# define OMAP_INT_1610_GPIO_BANK2 40 | |
| 213 | +# define OMAP_INT_1610_GPIO_BANK3 41 | |
| 214 | +# define OMAP_INT_1610_MMC2 42 | |
| 215 | +# define OMAP_INT_1610_CF 43 | |
| 216 | +# define OMAP_INT_1610_WAKE_UP_REQ 46 | |
| 217 | +# define OMAP_INT_1610_GPIO_BANK4 48 | |
| 218 | +# define OMAP_INT_1610_SPI 49 | |
| 219 | +# define OMAP_INT_1610_DMA_CH6 53 | |
| 220 | +# define OMAP_INT_1610_DMA_CH7 54 | |
| 221 | +# define OMAP_INT_1610_DMA_CH8 55 | |
| 222 | +# define OMAP_INT_1610_DMA_CH9 56 | |
| 223 | +# define OMAP_INT_1610_DMA_CH10 57 | |
| 224 | +# define OMAP_INT_1610_DMA_CH11 58 | |
| 225 | +# define OMAP_INT_1610_DMA_CH12 59 | |
| 226 | +# define OMAP_INT_1610_DMA_CH13 60 | |
| 227 | +# define OMAP_INT_1610_DMA_CH14 61 | |
| 228 | +# define OMAP_INT_1610_DMA_CH15 62 | |
| 229 | +# define OMAP_INT_1610_NAND 63 | |
| 230 | + | |
| 231 | +/* | |
| 232 | + * OMAP-730 specific IRQ numbers for level 2 interrupt handler | |
| 233 | + */ | |
| 234 | +# define OMAP_INT_730_HW_ERRORS 0 | |
| 235 | +# define OMAP_INT_730_NFIQ_PWR_FAIL 1 | |
| 236 | +# define OMAP_INT_730_CFCD 2 | |
| 237 | +# define OMAP_INT_730_CFIREQ 3 | |
| 238 | +# define OMAP_INT_730_I2C 4 | |
| 239 | +# define OMAP_INT_730_PCC 5 | |
| 240 | +# define OMAP_INT_730_MPU_EXT_NIRQ 6 | |
| 241 | +# define OMAP_INT_730_SPI_100K_1 7 | |
| 242 | +# define OMAP_INT_730_SYREN_SPI 8 | |
| 243 | +# define OMAP_INT_730_VLYNQ 9 | |
| 244 | +# define OMAP_INT_730_GPIO_BANK4 10 | |
| 245 | +# define OMAP_INT_730_McBSP1TX 11 | |
| 246 | +# define OMAP_INT_730_McBSP1RX 12 | |
| 247 | +# define OMAP_INT_730_McBSP1RX_OF 13 | |
| 248 | +# define OMAP_INT_730_UART_MODEM_IRDA_2 14 | |
| 249 | +# define OMAP_INT_730_UART_MODEM_1 15 | |
| 250 | +# define OMAP_INT_730_MCSI 16 | |
| 251 | +# define OMAP_INT_730_uWireTX 17 | |
| 252 | +# define OMAP_INT_730_uWireRX 18 | |
| 253 | +# define OMAP_INT_730_SMC_CD 19 | |
| 254 | +# define OMAP_INT_730_SMC_IREQ 20 | |
| 255 | +# define OMAP_INT_730_HDQ_1WIRE 21 | |
| 256 | +# define OMAP_INT_730_TIMER32K 22 | |
| 257 | +# define OMAP_INT_730_MMC_SDIO 23 | |
| 258 | +# define OMAP_INT_730_UPLD 24 | |
| 259 | +# define OMAP_INT_730_USB_HHC_1 27 | |
| 260 | +# define OMAP_INT_730_USB_HHC_2 28 | |
| 261 | +# define OMAP_INT_730_USB_GENI 29 | |
| 262 | +# define OMAP_INT_730_USB_OTG 30 | |
| 263 | +# define OMAP_INT_730_CAMERA_IF 31 | |
| 264 | +# define OMAP_INT_730_RNG 32 | |
| 265 | +# define OMAP_INT_730_DUAL_MODE_TIMER 33 | |
| 266 | +# define OMAP_INT_730_DBB_RF_EN 34 | |
| 267 | +# define OMAP_INT_730_MPUIO_KEYPAD 35 | |
| 268 | +# define OMAP_INT_730_SHA1_MD5 36 | |
| 269 | +# define OMAP_INT_730_SPI_100K_2 37 | |
| 270 | +# define OMAP_INT_730_RNG_IDLE 38 | |
| 271 | +# define OMAP_INT_730_MPUIO 39 | |
| 272 | +# define OMAP_INT_730_LLPC_LCD_CTRL_OFF 40 | |
| 273 | +# define OMAP_INT_730_LLPC_OE_FALLING 41 | |
| 274 | +# define OMAP_INT_730_LLPC_OE_RISING 42 | |
| 275 | +# define OMAP_INT_730_LLPC_VSYNC 43 | |
| 276 | +# define OMAP_INT_730_WAKE_UP_REQ 46 | |
| 277 | +# define OMAP_INT_730_DMA_CH6 53 | |
| 278 | +# define OMAP_INT_730_DMA_CH7 54 | |
| 279 | +# define OMAP_INT_730_DMA_CH8 55 | |
| 280 | +# define OMAP_INT_730_DMA_CH9 56 | |
| 281 | +# define OMAP_INT_730_DMA_CH10 57 | |
| 282 | +# define OMAP_INT_730_DMA_CH11 58 | |
| 283 | +# define OMAP_INT_730_DMA_CH12 59 | |
| 284 | +# define OMAP_INT_730_DMA_CH13 60 | |
| 285 | +# define OMAP_INT_730_DMA_CH14 61 | |
| 286 | +# define OMAP_INT_730_DMA_CH15 62 | |
| 287 | +# define OMAP_INT_730_NAND 63 | |
| 288 | + | |
| 289 | +/* | |
| 290 | + * OMAP-24xx common IRQ numbers | |
| 291 | + */ | |
| 292 | +# define OMAP_INT_24XX_SYS_NIRQ 7 | |
| 293 | +# define OMAP_INT_24XX_SDMA_IRQ0 12 | |
| 294 | +# define OMAP_INT_24XX_SDMA_IRQ1 13 | |
| 295 | +# define OMAP_INT_24XX_SDMA_IRQ2 14 | |
| 296 | +# define OMAP_INT_24XX_SDMA_IRQ3 15 | |
| 297 | +# define OMAP_INT_24XX_CAM_IRQ 24 | |
| 298 | +# define OMAP_INT_24XX_DSS_IRQ 25 | |
| 299 | +# define OMAP_INT_24XX_MAIL_U0_MPU 26 | |
| 300 | +# define OMAP_INT_24XX_DSP_UMA 27 | |
| 301 | +# define OMAP_INT_24XX_DSP_MMU 28 | |
| 302 | +# define OMAP_INT_24XX_GPIO_BANK1 29 | |
| 303 | +# define OMAP_INT_24XX_GPIO_BANK2 30 | |
| 304 | +# define OMAP_INT_24XX_GPIO_BANK3 31 | |
| 305 | +# define OMAP_INT_24XX_GPIO_BANK4 32 | |
| 306 | +# define OMAP_INT_24XX_GPIO_BANK5 33 | |
| 307 | +# define OMAP_INT_24XX_MAIL_U3_MPU 34 | |
| 308 | +# define OMAP_INT_24XX_GPTIMER1 37 | |
| 309 | +# define OMAP_INT_24XX_GPTIMER2 38 | |
| 310 | +# define OMAP_INT_24XX_GPTIMER3 39 | |
| 311 | +# define OMAP_INT_24XX_GPTIMER4 40 | |
| 312 | +# define OMAP_INT_24XX_GPTIMER5 41 | |
| 313 | +# define OMAP_INT_24XX_GPTIMER6 42 | |
| 314 | +# define OMAP_INT_24XX_GPTIMER7 43 | |
| 315 | +# define OMAP_INT_24XX_GPTIMER8 44 | |
| 316 | +# define OMAP_INT_24XX_GPTIMER9 45 | |
| 317 | +# define OMAP_INT_24XX_GPTIMER10 46 | |
| 318 | +# define OMAP_INT_24XX_GPTIMER11 47 | |
| 319 | +# define OMAP_INT_24XX_GPTIMER12 48 | |
| 320 | +# define OMAP_INT_24XX_MCBSP1_IRQ_TX 59 | |
| 321 | +# define OMAP_INT_24XX_MCBSP1_IRQ_RX 60 | |
| 322 | +# define OMAP_INT_24XX_MCBSP2_IRQ_TX 62 | |
| 323 | +# define OMAP_INT_24XX_MCBSP2_IRQ_RX 63 | |
| 324 | +# define OMAP_INT_24XX_UART1_IRQ 72 | |
| 325 | +# define OMAP_INT_24XX_UART2_IRQ 73 | |
| 326 | +# define OMAP_INT_24XX_UART3_IRQ 74 | |
| 327 | +# define OMAP_INT_24XX_USB_IRQ_GEN 75 | |
| 328 | +# define OMAP_INT_24XX_USB_IRQ_NISO 76 | |
| 329 | +# define OMAP_INT_24XX_USB_IRQ_ISO 77 | |
| 330 | +# define OMAP_INT_24XX_USB_IRQ_HGEN 78 | |
| 331 | +# define OMAP_INT_24XX_USB_IRQ_HSOF 79 | |
| 332 | +# define OMAP_INT_24XX_USB_IRQ_OTG 80 | |
| 333 | +# define OMAP_INT_24XX_MMC_IRQ 83 | |
| 334 | +# define OMAP_INT_243X_HS_USB_MC 92 | |
| 335 | +# define OMAP_INT_243X_HS_USB_DMA 93 | |
| 336 | +# define OMAP_INT_243X_CARKIT 94 | |
| 337 | + | |
| 338 | +struct omap_dma_s; | |
| 339 | +struct omap_dma_s *omap_dma_init(target_phys_addr_t base, | |
| 340 | + qemu_irq pic[], struct omap_mpu_state_s *mpu, omap_clk clk); | |
| 341 | + | |
| 342 | +enum omap_dma_port { | |
| 343 | + emiff = 0, | |
| 344 | + emifs, | |
| 345 | + imif, | |
| 346 | + tipb, | |
| 347 | + local, | |
| 348 | + tipb_mpui, | |
| 349 | + omap_dma_port_last, | |
| 350 | +}; | |
| 351 | + | |
| 352 | +struct omap_dma_lcd_channel_s { | |
| 353 | + enum omap_dma_port src; | |
| 354 | + target_phys_addr_t src_f1_top; | |
| 355 | + target_phys_addr_t src_f1_bottom; | |
| 356 | + target_phys_addr_t src_f2_top; | |
| 357 | + target_phys_addr_t src_f2_bottom; | |
| 358 | + /* Destination port is fixed. */ | |
| 359 | + int interrupts; | |
| 360 | + int condition; | |
| 361 | + int dual; | |
| 362 | + | |
| 363 | + int current_frame; | |
| 364 | + ram_addr_t phys_framebuffer[2]; | |
| 365 | + qemu_irq irq; | |
| 366 | + struct omap_mpu_state_s *mpu; | |
| 367 | +}; | |
| 368 | + | |
| 369 | +/* | |
| 370 | + * DMA request numbers for OMAP1 | |
| 371 | + * See /usr/include/asm-arm/arch-omap/dma.h in Linux. | |
| 372 | + */ | |
| 373 | +# define OMAP_DMA_NO_DEVICE 0 | |
| 374 | +# define OMAP_DMA_MCSI1_TX 1 | |
| 375 | +# define OMAP_DMA_MCSI1_RX 2 | |
| 376 | +# define OMAP_DMA_I2C_RX 3 | |
| 377 | +# define OMAP_DMA_I2C_TX 4 | |
| 378 | +# define OMAP_DMA_EXT_NDMA_REQ0 5 | |
| 379 | +# define OMAP_DMA_EXT_NDMA_REQ1 6 | |
| 380 | +# define OMAP_DMA_UWIRE_TX 7 | |
| 381 | +# define OMAP_DMA_MCBSP1_TX 8 | |
| 382 | +# define OMAP_DMA_MCBSP1_RX 9 | |
| 383 | +# define OMAP_DMA_MCBSP3_TX 10 | |
| 384 | +# define OMAP_DMA_MCBSP3_RX 11 | |
| 385 | +# define OMAP_DMA_UART1_TX 12 | |
| 386 | +# define OMAP_DMA_UART1_RX 13 | |
| 387 | +# define OMAP_DMA_UART2_TX 14 | |
| 388 | +# define OMAP_DMA_UART2_RX 15 | |
| 389 | +# define OMAP_DMA_MCBSP2_TX 16 | |
| 390 | +# define OMAP_DMA_MCBSP2_RX 17 | |
| 391 | +# define OMAP_DMA_UART3_TX 18 | |
| 392 | +# define OMAP_DMA_UART3_RX 19 | |
| 393 | +# define OMAP_DMA_CAMERA_IF_RX 20 | |
| 394 | +# define OMAP_DMA_MMC_TX 21 | |
| 395 | +# define OMAP_DMA_MMC_RX 22 | |
| 396 | +# define OMAP_DMA_NAND 23 /* Not in OMAP310 */ | |
| 397 | +# define OMAP_DMA_IRQ_LCD_LINE 24 /* Not in OMAP310 */ | |
| 398 | +# define OMAP_DMA_MEMORY_STICK 25 /* Not in OMAP310 */ | |
| 399 | +# define OMAP_DMA_USB_W2FC_RX0 26 | |
| 400 | +# define OMAP_DMA_USB_W2FC_RX1 27 | |
| 401 | +# define OMAP_DMA_USB_W2FC_RX2 28 | |
| 402 | +# define OMAP_DMA_USB_W2FC_TX0 29 | |
| 403 | +# define OMAP_DMA_USB_W2FC_TX1 30 | |
| 404 | +# define OMAP_DMA_USB_W2FC_TX2 31 | |
| 405 | + | |
| 406 | +/* These are only for 1610 */ | |
| 407 | +# define OMAP_DMA_CRYPTO_DES_IN 32 | |
| 408 | +# define OMAP_DMA_SPI_TX 33 | |
| 409 | +# define OMAP_DMA_SPI_RX 34 | |
| 410 | +# define OMAP_DMA_CRYPTO_HASH 35 | |
| 411 | +# define OMAP_DMA_CCP_ATTN 36 | |
| 412 | +# define OMAP_DMA_CCP_FIFO_NOT_EMPTY 37 | |
| 413 | +# define OMAP_DMA_CMT_APE_TX_CHAN_0 38 | |
| 414 | +# define OMAP_DMA_CMT_APE_RV_CHAN_0 39 | |
| 415 | +# define OMAP_DMA_CMT_APE_TX_CHAN_1 40 | |
| 416 | +# define OMAP_DMA_CMT_APE_RV_CHAN_1 41 | |
| 417 | +# define OMAP_DMA_CMT_APE_TX_CHAN_2 42 | |
| 418 | +# define OMAP_DMA_CMT_APE_RV_CHAN_2 43 | |
| 419 | +# define OMAP_DMA_CMT_APE_TX_CHAN_3 44 | |
| 420 | +# define OMAP_DMA_CMT_APE_RV_CHAN_3 45 | |
| 421 | +# define OMAP_DMA_CMT_APE_TX_CHAN_4 46 | |
| 422 | +# define OMAP_DMA_CMT_APE_RV_CHAN_4 47 | |
| 423 | +# define OMAP_DMA_CMT_APE_TX_CHAN_5 48 | |
| 424 | +# define OMAP_DMA_CMT_APE_RV_CHAN_5 49 | |
| 425 | +# define OMAP_DMA_CMT_APE_TX_CHAN_6 50 | |
| 426 | +# define OMAP_DMA_CMT_APE_RV_CHAN_6 51 | |
| 427 | +# define OMAP_DMA_CMT_APE_TX_CHAN_7 52 | |
| 428 | +# define OMAP_DMA_CMT_APE_RV_CHAN_7 53 | |
| 429 | +# define OMAP_DMA_MMC2_TX 54 | |
| 430 | +# define OMAP_DMA_MMC2_RX 55 | |
| 431 | +# define OMAP_DMA_CRYPTO_DES_OUT 56 | |
| 432 | + | |
| 433 | +struct omap_mpu_timer_s; | |
| 434 | +struct omap_mpu_timer_s *omap_mpu_timer_init(target_phys_addr_t base, | |
| 435 | + qemu_irq irq, omap_clk clk); | |
| 436 | + | |
| 437 | +struct omap_watchdog_timer_s; | |
| 438 | +struct omap_watchdog_timer_s *omap_wd_timer_init(target_phys_addr_t base, | |
| 439 | + qemu_irq irq, omap_clk clk); | |
| 440 | + | |
| 441 | +struct omap_32khz_timer_s; | |
| 442 | +struct omap_32khz_timer_s *omap_os_timer_init(target_phys_addr_t base, | |
| 443 | + qemu_irq irq, omap_clk clk); | |
| 444 | + | |
| 445 | +struct omap_tipb_bridge_s; | |
| 446 | +struct omap_tipb_bridge_s *omap_tipb_bridge_init(target_phys_addr_t base, | |
| 447 | + qemu_irq abort_irq, omap_clk clk); | |
| 448 | + | |
| 449 | +struct omap_uart_s; | |
| 450 | +struct omap_uart_s *omap_uart_init(target_phys_addr_t base, | |
| 451 | + qemu_irq irq, omap_clk clk, CharDriverState *chr); | |
| 452 | + | |
| 453 | +/* omap_lcdc.c */ | |
| 454 | +struct omap_lcd_panel_s; | |
| 455 | +void omap_lcdc_reset(struct omap_lcd_panel_s *s); | |
| 456 | +struct omap_lcd_panel_s *omap_lcdc_init(target_phys_addr_t base, qemu_irq irq, | |
| 457 | + struct omap_dma_lcd_channel_s *dma, DisplayState *ds, | |
| 458 | + ram_addr_t imif_base, ram_addr_t emiff_base, omap_clk clk); | |
| 459 | + | |
| 460 | +# define cpu_is_omap310(cpu) (cpu->mpu_model == omap310) | |
| 461 | +# define cpu_is_omap1510(cpu) (cpu->mpu_model == omap1510) | |
| 462 | +# define cpu_is_omap15xx(cpu) \ | |
| 463 | + (cpu_is_omap310(cpu) || cpu_is_omap1510(cpu)) | |
| 464 | +# define cpu_class_omap1(cpu) 1 | |
| 465 | + | |
| 466 | +struct omap_mpu_state_s { | |
| 467 | + enum omap1_mpu_model { | |
| 468 | + omap310, | |
| 469 | + omap1510, | |
| 470 | + } mpu_model; | |
| 471 | + | |
| 472 | + CPUState *env; | |
| 473 | + | |
| 474 | + qemu_irq *irq[2]; | |
| 475 | + qemu_irq *drq; | |
| 476 | + | |
| 477 | + qemu_irq wakeup; | |
| 478 | + | |
| 479 | + struct omap_dma_port_if_s { | |
| 480 | + uint32_t (*read[3])(struct omap_mpu_state_s *s, | |
| 481 | + target_phys_addr_t offset); | |
| 482 | + void (*write[3])(struct omap_mpu_state_s *s, | |
| 483 | + target_phys_addr_t offset, uint32_t value); | |
| 484 | + int (*addr_valid)(struct omap_mpu_state_s *s, | |
| 485 | + target_phys_addr_t addr); | |
| 486 | + } port[omap_dma_port_last]; | |
| 487 | + | |
| 488 | + unsigned long sdram_size; | |
| 489 | + unsigned long sram_size; | |
| 490 | + | |
| 491 | + /* MPUI-TIPB peripherals */ | |
| 492 | + struct omap_uart_s *uart3; | |
| 493 | + | |
| 494 | + /* MPU public TIPB peripherals */ | |
| 495 | + struct omap_32khz_timer_s *os_timer; | |
| 496 | + | |
| 497 | + struct omap_uart_s *uart1; | |
| 498 | + struct omap_uart_s *uart2; | |
| 499 | + | |
| 500 | + /* MPU private TIPB peripherals */ | |
| 501 | + struct omap_intr_handler_s *ih[2]; | |
| 502 | + | |
| 503 | + struct omap_dma_s *dma; | |
| 504 | + | |
| 505 | + struct omap_mpu_timer_s *timer[3]; | |
| 506 | + struct omap_watchdog_timer_s *wdt; | |
| 507 | + | |
| 508 | + struct omap_lcd_panel_s *lcd; | |
| 509 | + | |
| 510 | + target_phys_addr_t ulpd_pm_base; | |
| 511 | + uint32_t ulpd_pm_regs[21]; | |
| 512 | + int64_t ulpd_gauge_start; | |
| 513 | + | |
| 514 | + target_phys_addr_t pin_cfg_base; | |
| 515 | + uint32_t func_mux_ctrl[14]; | |
| 516 | + uint32_t comp_mode_ctrl[1]; | |
| 517 | + uint32_t pull_dwn_ctrl[4]; | |
| 518 | + uint32_t gate_inh_ctrl[1]; | |
| 519 | + uint32_t voltage_ctrl[1]; | |
| 520 | + uint32_t test_dbg_ctrl[1]; | |
| 521 | + uint32_t mod_conf_ctrl[1]; | |
| 522 | + int compat1509; | |
| 523 | + | |
| 524 | + uint32_t mpui_ctrl; | |
| 525 | + target_phys_addr_t mpui_base; | |
| 526 | + | |
| 527 | + struct omap_tipb_bridge_s *private_tipb; | |
| 528 | + struct omap_tipb_bridge_s *public_tipb; | |
| 529 | + | |
| 530 | + target_phys_addr_t tcmi_base; | |
| 531 | + uint32_t tcmi_regs[17]; | |
| 532 | + | |
| 533 | + struct dpll_ctl_s { | |
| 534 | + target_phys_addr_t base; | |
| 535 | + uint16_t mode; | |
| 536 | + omap_clk dpll; | |
| 537 | + } dpll[3]; | |
| 538 | + | |
| 539 | + omap_clk clks; | |
| 540 | + struct { | |
| 541 | + target_phys_addr_t mpu_base; | |
| 542 | + target_phys_addr_t dsp_base; | |
| 543 | + | |
| 544 | + int cold_start; | |
| 545 | + int clocking_scheme; | |
| 546 | + uint16_t arm_ckctl; | |
| 547 | + uint16_t arm_idlect1; | |
| 548 | + uint16_t arm_idlect2; | |
| 549 | + uint16_t arm_ewupct; | |
| 550 | + uint16_t arm_rstct1; | |
| 551 | + uint16_t arm_rstct2; | |
| 552 | + uint16_t arm_ckout1; | |
| 553 | + int dpll1_mode; | |
| 554 | + uint16_t dsp_idlect1; | |
| 555 | + uint16_t dsp_idlect2; | |
| 556 | + uint16_t dsp_rstct2; | |
| 557 | + } clkm; | |
| 558 | +} *omap310_mpu_init(unsigned long sdram_size, | |
| 559 | + DisplayState *ds, const char *core); | |
| 560 | + | |
| 561 | +# if TARGET_PHYS_ADDR_BITS == 32 | |
| 562 | +# define OMAP_FMT_plx "%#08x" | |
| 563 | +# elif TARGET_PHYS_ADDR_BITS == 64 | |
| 564 | +# define OMAP_FMT_plx "%#08" PRIx64 | |
| 565 | +# else | |
| 566 | +# error TARGET_PHYS_ADDR_BITS undefined | |
| 567 | +# endif | |
| 568 | + | |
| 569 | +# define OMAP_BAD_REG(paddr) \ | |
| 570 | + printf("%s: Bad register " OMAP_FMT_plx "\n", __FUNCTION__, paddr) | |
| 571 | +# define OMAP_RO_REG(paddr) \ | |
| 572 | + printf("%s: Read-only register " OMAP_FMT_plx "\n", \ | |
| 573 | + __FUNCTION__, paddr) | |
| 574 | +# define OMAP_16B_REG(paddr) \ | |
| 575 | + printf("%s: 16-bit register " OMAP_FMT_plx "\n", \ | |
| 576 | + __FUNCTION__, paddr) | |
| 577 | +# define OMAP_32B_REG(paddr) \ | |
| 578 | + printf("%s: 32-bit register " OMAP_FMT_plx "\n", \ | |
| 579 | + __FUNCTION__, paddr) | |
| 580 | + | |
| 581 | +#endif /* hw_omap_h */ | ... | ... |
hw/omap1_clk.c
0 → 100644
| 1 | +/* | |
| 2 | + * OMAP clocks. | |
| 3 | + * | |
| 4 | + * Copyright (C) 2006-2007 Andrzej Zaborowski <balrog@zabor.org> | |
| 5 | + * | |
| 6 | + * Clocks data comes in part from arch/arm/mach-omap1/clock.h in Linux. | |
| 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 of | |
| 11 | + * the License, or (at your option) any later version. | |
| 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 | +#include "vl.h" | |
| 24 | + | |
| 25 | +struct clk { | |
| 26 | + const char *name; | |
| 27 | + const char *alias; | |
| 28 | + struct clk *parent; | |
| 29 | + struct clk *child1; | |
| 30 | + struct clk *sibling; | |
| 31 | +#define ALWAYS_ENABLED (1 << 0) | |
| 32 | +#define CLOCK_IN_OMAP310 (1 << 10) | |
| 33 | +#define CLOCK_IN_OMAP730 (1 << 11) | |
| 34 | +#define CLOCK_IN_OMAP1510 (1 << 12) | |
| 35 | +#define CLOCK_IN_OMAP16XX (1 << 13) | |
| 36 | + uint32_t flags; | |
| 37 | + int id; | |
| 38 | + | |
| 39 | + int running; /* Is currently ticking */ | |
| 40 | + int enabled; /* Is enabled, regardless of its input clk */ | |
| 41 | + unsigned long rate; /* Current rate (if .running) */ | |
| 42 | + unsigned int divisor; /* Rate relative to input (if .enabled) */ | |
| 43 | + unsigned int multiplier; /* Rate relative to input (if .enabled) */ | |
| 44 | + qemu_irq users[16]; /* Who to notify on change */ | |
| 45 | + int usecount; /* Automatically idle when unused */ | |
| 46 | +}; | |
| 47 | + | |
| 48 | +static struct clk xtal_osc12m = { | |
| 49 | + .name = "xtal_osc_12m", | |
| 50 | + .rate = 12000000, | |
| 51 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310, | |
| 52 | +}; | |
| 53 | + | |
| 54 | +static struct clk xtal_osc32k = { | |
| 55 | + .name = "xtal_osc_32k", | |
| 56 | + .rate = 32768, | |
| 57 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310, | |
| 58 | +}; | |
| 59 | + | |
| 60 | +static struct clk ck_ref = { | |
| 61 | + .name = "ck_ref", | |
| 62 | + .alias = "clkin", | |
| 63 | + .parent = &xtal_osc12m, | |
| 64 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310 | | |
| 65 | + ALWAYS_ENABLED, | |
| 66 | +}; | |
| 67 | + | |
| 68 | +/* If a dpll is disabled it becomes a bypass, child clocks don't stop */ | |
| 69 | +static struct clk dpll1 = { | |
| 70 | + .name = "dpll1", | |
| 71 | + .parent = &ck_ref, | |
| 72 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310 | | |
| 73 | + ALWAYS_ENABLED, | |
| 74 | +}; | |
| 75 | + | |
| 76 | +static struct clk dpll2 = { | |
| 77 | + .name = "dpll2", | |
| 78 | + .parent = &ck_ref, | |
| 79 | + .flags = CLOCK_IN_OMAP310 | ALWAYS_ENABLED, | |
| 80 | +}; | |
| 81 | + | |
| 82 | +static struct clk dpll3 = { | |
| 83 | + .name = "dpll3", | |
| 84 | + .parent = &ck_ref, | |
| 85 | + .flags = CLOCK_IN_OMAP310 | ALWAYS_ENABLED, | |
| 86 | +}; | |
| 87 | + | |
| 88 | +static struct clk dpll4 = { | |
| 89 | + .name = "dpll4", | |
| 90 | + .parent = &ck_ref, | |
| 91 | + .multiplier = 4, | |
| 92 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310, | |
| 93 | +}; | |
| 94 | + | |
| 95 | +static struct clk apll = { | |
| 96 | + .name = "apll", | |
| 97 | + .parent = &ck_ref, | |
| 98 | + .multiplier = 48, | |
| 99 | + .divisor = 12, | |
| 100 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310, | |
| 101 | +}; | |
| 102 | + | |
| 103 | +static struct clk ck_48m = { | |
| 104 | + .name = "ck_48m", | |
| 105 | + .parent = &dpll4, /* either dpll4 or apll */ | |
| 106 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310, | |
| 107 | +}; | |
| 108 | + | |
| 109 | +static struct clk ck_dpll1out = { | |
| 110 | + .name = "ck_dpll1out", | |
| 111 | + .parent = &dpll1, | |
| 112 | + .flags = CLOCK_IN_OMAP16XX, | |
| 113 | +}; | |
| 114 | + | |
| 115 | +static struct clk sossi_ck = { | |
| 116 | + .name = "ck_sossi", | |
| 117 | + .parent = &ck_dpll1out, | |
| 118 | + .flags = CLOCK_IN_OMAP16XX, | |
| 119 | +}; | |
| 120 | + | |
| 121 | +static struct clk clkm1 = { | |
| 122 | + .name = "clkm1", | |
| 123 | + .alias = "ck_gen1", | |
| 124 | + .parent = &dpll1, | |
| 125 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310 | | |
| 126 | + ALWAYS_ENABLED, | |
| 127 | +}; | |
| 128 | + | |
| 129 | +static struct clk clkm2 = { | |
| 130 | + .name = "clkm2", | |
| 131 | + .alias = "ck_gen2", | |
| 132 | + .parent = &dpll1, | |
| 133 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310 | | |
| 134 | + ALWAYS_ENABLED, | |
| 135 | +}; | |
| 136 | + | |
| 137 | +static struct clk clkm3 = { | |
| 138 | + .name = "clkm3", | |
| 139 | + .alias = "ck_gen3", | |
| 140 | + .parent = &dpll1, /* either dpll1 or ck_ref */ | |
| 141 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310 | | |
| 142 | + ALWAYS_ENABLED, | |
| 143 | +}; | |
| 144 | + | |
| 145 | +static struct clk arm_ck = { | |
| 146 | + .name = "arm_ck", | |
| 147 | + .alias = "mpu_ck", | |
| 148 | + .parent = &clkm1, | |
| 149 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310 | | |
| 150 | + ALWAYS_ENABLED, | |
| 151 | +}; | |
| 152 | + | |
| 153 | +static struct clk armper_ck = { | |
| 154 | + .name = "armper_ck", | |
| 155 | + .alias = "mpuper_ck", | |
| 156 | + .parent = &clkm1, | |
| 157 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310, | |
| 158 | +}; | |
| 159 | + | |
| 160 | +static struct clk arm_gpio_ck = { | |
| 161 | + .name = "arm_gpio_ck", | |
| 162 | + .alias = "mpu_gpio_ck", | |
| 163 | + .parent = &clkm1, | |
| 164 | + .divisor = 1, | |
| 165 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP310, | |
| 166 | +}; | |
| 167 | + | |
| 168 | +static struct clk armxor_ck = { | |
| 169 | + .name = "armxor_ck", | |
| 170 | + .alias = "mpuxor_ck", | |
| 171 | + .parent = &ck_ref, | |
| 172 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310, | |
| 173 | +}; | |
| 174 | + | |
| 175 | +static struct clk armtim_ck = { | |
| 176 | + .name = "armtim_ck", | |
| 177 | + .alias = "mputim_ck", | |
| 178 | + .parent = &ck_ref, /* either CLKIN or DPLL1 */ | |
| 179 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310, | |
| 180 | +}; | |
| 181 | + | |
| 182 | +static struct clk armwdt_ck = { | |
| 183 | + .name = "armwdt_ck", | |
| 184 | + .alias = "mpuwd_ck", | |
| 185 | + .parent = &clkm1, | |
| 186 | + .divisor = 14, | |
| 187 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310 | | |
| 188 | + ALWAYS_ENABLED, | |
| 189 | +}; | |
| 190 | + | |
| 191 | +static struct clk arminth_ck16xx = { | |
| 192 | + .name = "arminth_ck", | |
| 193 | + .parent = &arm_ck, | |
| 194 | + .flags = CLOCK_IN_OMAP16XX | ALWAYS_ENABLED, | |
| 195 | + /* Note: On 16xx the frequency can be divided by 2 by programming | |
| 196 | + * ARM_CKCTL:ARM_INTHCK_SEL(14) to 1 | |
| 197 | + * | |
| 198 | + * 1510 version is in TC clocks. | |
| 199 | + */ | |
| 200 | +}; | |
| 201 | + | |
| 202 | +static struct clk dsp_ck = { | |
| 203 | + .name = "dsp_ck", | |
| 204 | + .parent = &clkm2, | |
| 205 | + .flags = CLOCK_IN_OMAP310 | CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX, | |
| 206 | +}; | |
| 207 | + | |
| 208 | +static struct clk dspmmu_ck = { | |
| 209 | + .name = "dspmmu_ck", | |
| 210 | + .parent = &clkm2, | |
| 211 | + .flags = CLOCK_IN_OMAP310 | CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | | |
| 212 | + ALWAYS_ENABLED, | |
| 213 | +}; | |
| 214 | + | |
| 215 | +static struct clk dspper_ck = { | |
| 216 | + .name = "dspper_ck", | |
| 217 | + .parent = &clkm2, | |
| 218 | + .flags = CLOCK_IN_OMAP310 | CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX, | |
| 219 | +}; | |
| 220 | + | |
| 221 | +static struct clk dspxor_ck = { | |
| 222 | + .name = "dspxor_ck", | |
| 223 | + .parent = &ck_ref, | |
| 224 | + .flags = CLOCK_IN_OMAP310 | CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX, | |
| 225 | +}; | |
| 226 | + | |
| 227 | +static struct clk dsptim_ck = { | |
| 228 | + .name = "dsptim_ck", | |
| 229 | + .parent = &ck_ref, | |
| 230 | + .flags = CLOCK_IN_OMAP310 | CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX, | |
| 231 | +}; | |
| 232 | + | |
| 233 | +static struct clk tc_ck = { | |
| 234 | + .name = "tc_ck", | |
| 235 | + .parent = &clkm3, | |
| 236 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | | |
| 237 | + CLOCK_IN_OMAP730 | CLOCK_IN_OMAP310 | | |
| 238 | + ALWAYS_ENABLED, | |
| 239 | +}; | |
| 240 | + | |
| 241 | +static struct clk arminth_ck15xx = { | |
| 242 | + .name = "arminth_ck", | |
| 243 | + .parent = &tc_ck, | |
| 244 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP310 | ALWAYS_ENABLED, | |
| 245 | + /* Note: On 1510 the frequency follows TC_CK | |
| 246 | + * | |
| 247 | + * 16xx version is in MPU clocks. | |
| 248 | + */ | |
| 249 | +}; | |
| 250 | + | |
| 251 | +static struct clk tipb_ck = { | |
| 252 | + /* No-idle controlled by "tc_ck" */ | |
| 253 | + .name = "tipb_ck", | |
| 254 | + .parent = &tc_ck, | |
| 255 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP310 | ALWAYS_ENABLED, | |
| 256 | +}; | |
| 257 | + | |
| 258 | +static struct clk l3_ocpi_ck = { | |
| 259 | + /* No-idle controlled by "tc_ck" */ | |
| 260 | + .name = "l3_ocpi_ck", | |
| 261 | + .parent = &tc_ck, | |
| 262 | + .flags = CLOCK_IN_OMAP16XX, | |
| 263 | +}; | |
| 264 | + | |
| 265 | +static struct clk tc1_ck = { | |
| 266 | + .name = "tc1_ck", | |
| 267 | + .parent = &tc_ck, | |
| 268 | + .flags = CLOCK_IN_OMAP16XX, | |
| 269 | +}; | |
| 270 | + | |
| 271 | +static struct clk tc2_ck = { | |
| 272 | + .name = "tc2_ck", | |
| 273 | + .parent = &tc_ck, | |
| 274 | + .flags = CLOCK_IN_OMAP16XX, | |
| 275 | +}; | |
| 276 | + | |
| 277 | +static struct clk dma_ck = { | |
| 278 | + /* No-idle controlled by "tc_ck" */ | |
| 279 | + .name = "dma_ck", | |
| 280 | + .parent = &tc_ck, | |
| 281 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310 | | |
| 282 | + ALWAYS_ENABLED, | |
| 283 | +}; | |
| 284 | + | |
| 285 | +static struct clk dma_lcdfree_ck = { | |
| 286 | + .name = "dma_lcdfree_ck", | |
| 287 | + .parent = &tc_ck, | |
| 288 | + .flags = CLOCK_IN_OMAP16XX | ALWAYS_ENABLED, | |
| 289 | +}; | |
| 290 | + | |
| 291 | +static struct clk api_ck = { | |
| 292 | + .name = "api_ck", | |
| 293 | + .alias = "mpui_ck", | |
| 294 | + .parent = &tc_ck, | |
| 295 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310, | |
| 296 | +}; | |
| 297 | + | |
| 298 | +static struct clk lb_ck = { | |
| 299 | + .name = "lb_ck", | |
| 300 | + .parent = &tc_ck, | |
| 301 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP310, | |
| 302 | +}; | |
| 303 | + | |
| 304 | +static struct clk lbfree_ck = { | |
| 305 | + .name = "lbfree_ck", | |
| 306 | + .parent = &tc_ck, | |
| 307 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP310, | |
| 308 | +}; | |
| 309 | + | |
| 310 | +static struct clk rhea1_ck = { | |
| 311 | + .name = "rhea1_ck", | |
| 312 | + .parent = &tc_ck, | |
| 313 | + .flags = CLOCK_IN_OMAP16XX | ALWAYS_ENABLED, | |
| 314 | +}; | |
| 315 | + | |
| 316 | +static struct clk rhea2_ck = { | |
| 317 | + .name = "rhea2_ck", | |
| 318 | + .parent = &tc_ck, | |
| 319 | + .flags = CLOCK_IN_OMAP16XX | ALWAYS_ENABLED, | |
| 320 | +}; | |
| 321 | + | |
| 322 | +static struct clk lcd_ck_16xx = { | |
| 323 | + .name = "lcd_ck", | |
| 324 | + .parent = &clkm3, | |
| 325 | + .flags = CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP730, | |
| 326 | +}; | |
| 327 | + | |
| 328 | +static struct clk lcd_ck_1510 = { | |
| 329 | + .name = "lcd_ck", | |
| 330 | + .parent = &clkm3, | |
| 331 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP310, | |
| 332 | +}; | |
| 333 | + | |
| 334 | +static struct clk uart1_1510 = { | |
| 335 | + .name = "uart1_ck", | |
| 336 | + /* Direct from ULPD, no real parent */ | |
| 337 | + .parent = &armper_ck, /* either armper_ck or dpll4 */ | |
| 338 | + .rate = 12000000, | |
| 339 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP310 | ALWAYS_ENABLED, | |
| 340 | +}; | |
| 341 | + | |
| 342 | +static struct clk uart1_16xx = { | |
| 343 | + .name = "uart1_ck", | |
| 344 | + /* Direct from ULPD, no real parent */ | |
| 345 | + .parent = &armper_ck, | |
| 346 | + .rate = 48000000, | |
| 347 | + .flags = CLOCK_IN_OMAP16XX, | |
| 348 | +}; | |
| 349 | + | |
| 350 | +static struct clk uart2_ck = { | |
| 351 | + .name = "uart2_ck", | |
| 352 | + /* Direct from ULPD, no real parent */ | |
| 353 | + .parent = &armper_ck, /* either armper_ck or dpll4 */ | |
| 354 | + .rate = 12000000, | |
| 355 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310 | | |
| 356 | + ALWAYS_ENABLED, | |
| 357 | +}; | |
| 358 | + | |
| 359 | +static struct clk uart3_1510 = { | |
| 360 | + .name = "uart3_ck", | |
| 361 | + /* Direct from ULPD, no real parent */ | |
| 362 | + .parent = &armper_ck,/* either armper_ck or dpll4 */ | |
| 363 | + .rate = 12000000, | |
| 364 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP310 | ALWAYS_ENABLED, | |
| 365 | +}; | |
| 366 | + | |
| 367 | +static struct clk uart3_16xx = { | |
| 368 | + .name = "uart3_ck", | |
| 369 | + /* Direct from ULPD, no real parent */ | |
| 370 | + .parent = &armper_ck, | |
| 371 | + .rate = 48000000, | |
| 372 | + .flags = CLOCK_IN_OMAP16XX, | |
| 373 | +}; | |
| 374 | + | |
| 375 | +static struct clk usb_clk0 = { /* 6 MHz output on W4_USB_CLK0 */ | |
| 376 | + .name = "usb_clk0", | |
| 377 | + .alias = "usb.clko", | |
| 378 | + /* Direct from ULPD, no parent */ | |
| 379 | + .rate = 6000000, | |
| 380 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310, | |
| 381 | +}; | |
| 382 | + | |
| 383 | +static struct clk usb_hhc_ck1510 = { | |
| 384 | + .name = "usb_hhc_ck", | |
| 385 | + /* Direct from ULPD, no parent */ | |
| 386 | + .rate = 48000000, /* Actually 2 clocks, 12MHz and 48MHz */ | |
| 387 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP310, | |
| 388 | +}; | |
| 389 | + | |
| 390 | +static struct clk usb_hhc_ck16xx = { | |
| 391 | + .name = "usb_hhc_ck", | |
| 392 | + /* Direct from ULPD, no parent */ | |
| 393 | + .rate = 48000000, | |
| 394 | + /* OTG_SYSCON_2.OTG_PADEN == 0 (not 1510-compatible) */ | |
| 395 | + .flags = CLOCK_IN_OMAP16XX, | |
| 396 | +}; | |
| 397 | + | |
| 398 | +static struct clk usb_dc_ck = { | |
| 399 | + .name = "usb_dc_ck", | |
| 400 | + /* Direct from ULPD, no parent */ | |
| 401 | + .rate = 48000000, | |
| 402 | + .flags = CLOCK_IN_OMAP16XX, | |
| 403 | +}; | |
| 404 | + | |
| 405 | +static struct clk mclk_1510 = { | |
| 406 | + .name = "mclk", | |
| 407 | + /* Direct from ULPD, no parent. May be enabled by ext hardware. */ | |
| 408 | + .rate = 12000000, | |
| 409 | + .flags = CLOCK_IN_OMAP1510, | |
| 410 | +}; | |
| 411 | + | |
| 412 | +static struct clk bclk_310 = { | |
| 413 | + .name = "bt_mclk_out", /* Alias midi_mclk_out? */ | |
| 414 | + .parent = &armper_ck, | |
| 415 | + .flags = CLOCK_IN_OMAP310, | |
| 416 | +}; | |
| 417 | + | |
| 418 | +static struct clk mclk_310 = { | |
| 419 | + .name = "com_mclk_out", | |
| 420 | + .parent = &armper_ck, | |
| 421 | + .flags = CLOCK_IN_OMAP310, | |
| 422 | +}; | |
| 423 | + | |
| 424 | +static struct clk mclk_16xx = { | |
| 425 | + .name = "mclk", | |
| 426 | + /* Direct from ULPD, no parent. May be enabled by ext hardware. */ | |
| 427 | + .flags = CLOCK_IN_OMAP16XX, | |
| 428 | +}; | |
| 429 | + | |
| 430 | +static struct clk bclk_1510 = { | |
| 431 | + .name = "bclk", | |
| 432 | + /* Direct from ULPD, no parent. May be enabled by ext hardware. */ | |
| 433 | + .rate = 12000000, | |
| 434 | + .flags = CLOCK_IN_OMAP1510, | |
| 435 | +}; | |
| 436 | + | |
| 437 | +static struct clk bclk_16xx = { | |
| 438 | + .name = "bclk", | |
| 439 | + /* Direct from ULPD, no parent. May be enabled by ext hardware. */ | |
| 440 | + .flags = CLOCK_IN_OMAP16XX, | |
| 441 | +}; | |
| 442 | + | |
| 443 | +static struct clk mmc1_ck = { | |
| 444 | + .name = "mmc_ck", | |
| 445 | + .id = 1, | |
| 446 | + /* Functional clock is direct from ULPD, interface clock is ARMPER */ | |
| 447 | + .parent = &armper_ck, /* either armper_ck or dpll4 */ | |
| 448 | + .rate = 48000000, | |
| 449 | + .flags = CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | CLOCK_IN_OMAP310, | |
| 450 | +}; | |
| 451 | + | |
| 452 | +static struct clk mmc2_ck = { | |
| 453 | + .name = "mmc_ck", | |
| 454 | + .id = 2, | |
| 455 | + /* Functional clock is direct from ULPD, interface clock is ARMPER */ | |
| 456 | + .parent = &armper_ck, | |
| 457 | + .rate = 48000000, | |
| 458 | + .flags = CLOCK_IN_OMAP16XX, | |
| 459 | +}; | |
| 460 | + | |
| 461 | +static struct clk cam_mclk = { | |
| 462 | + .name = "cam.mclk", | |
| 463 | + .flags = CLOCK_IN_OMAP310 | CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX, | |
| 464 | + .rate = 12000000, | |
| 465 | +}; | |
| 466 | + | |
| 467 | +static struct clk cam_exclk = { | |
| 468 | + .name = "cam.exclk", | |
| 469 | + .flags = CLOCK_IN_OMAP310 | CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX, | |
| 470 | + /* Either 12M from cam.mclk or 48M from dpll4 */ | |
| 471 | + .parent = &cam_mclk, | |
| 472 | +}; | |
| 473 | + | |
| 474 | +static struct clk cam_lclk = { | |
| 475 | + .name = "cam.lclk", | |
| 476 | + .flags = CLOCK_IN_OMAP310 | CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX, | |
| 477 | +}; | |
| 478 | + | |
| 479 | +static struct clk i2c_fck = { | |
| 480 | + .name = "i2c_fck", | |
| 481 | + .id = 1, | |
| 482 | + .flags = CLOCK_IN_OMAP310 | CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | | |
| 483 | + ALWAYS_ENABLED, | |
| 484 | + .parent = &armxor_ck, | |
| 485 | +}; | |
| 486 | + | |
| 487 | +static struct clk i2c_ick = { | |
| 488 | + .name = "i2c_ick", | |
| 489 | + .id = 1, | |
| 490 | + .flags = CLOCK_IN_OMAP16XX | ALWAYS_ENABLED, | |
| 491 | + .parent = &armper_ck, | |
| 492 | +}; | |
| 493 | + | |
| 494 | +static struct clk clk32k = { | |
| 495 | + .name = "clk32-kHz", | |
| 496 | + .flags = CLOCK_IN_OMAP310 | CLOCK_IN_OMAP1510 | CLOCK_IN_OMAP16XX | | |
| 497 | + ALWAYS_ENABLED, | |
| 498 | + .parent = &xtal_osc32k, | |
| 499 | +}; | |
| 500 | + | |
| 501 | +static struct clk *onchip_clks[] = { | |
| 502 | + /* non-ULPD clocks */ | |
| 503 | + &xtal_osc12m, | |
| 504 | + &xtal_osc32k, | |
| 505 | + &ck_ref, | |
| 506 | + &dpll1, | |
| 507 | + &dpll2, | |
| 508 | + &dpll3, | |
| 509 | + &dpll4, | |
| 510 | + &apll, | |
| 511 | + &ck_48m, | |
| 512 | + /* CK_GEN1 clocks */ | |
| 513 | + &clkm1, | |
| 514 | + &ck_dpll1out, | |
| 515 | + &sossi_ck, | |
| 516 | + &arm_ck, | |
| 517 | + &armper_ck, | |
| 518 | + &arm_gpio_ck, | |
| 519 | + &armxor_ck, | |
| 520 | + &armtim_ck, | |
| 521 | + &armwdt_ck, | |
| 522 | + &arminth_ck15xx, &arminth_ck16xx, | |
| 523 | + /* CK_GEN2 clocks */ | |
| 524 | + &clkm2, | |
| 525 | + &dsp_ck, | |
| 526 | + &dspmmu_ck, | |
| 527 | + &dspper_ck, | |
| 528 | + &dspxor_ck, | |
| 529 | + &dsptim_ck, | |
| 530 | + /* CK_GEN3 clocks */ | |
| 531 | + &clkm3, | |
| 532 | + &tc_ck, | |
| 533 | + &tipb_ck, | |
| 534 | + &l3_ocpi_ck, | |
| 535 | + &tc1_ck, | |
| 536 | + &tc2_ck, | |
| 537 | + &dma_ck, | |
| 538 | + &dma_lcdfree_ck, | |
| 539 | + &api_ck, | |
| 540 | + &lb_ck, | |
| 541 | + &lbfree_ck, | |
| 542 | + &rhea1_ck, | |
| 543 | + &rhea2_ck, | |
| 544 | + &lcd_ck_16xx, | |
| 545 | + &lcd_ck_1510, | |
| 546 | + /* ULPD clocks */ | |
| 547 | + &uart1_1510, | |
| 548 | + &uart1_16xx, | |
| 549 | + &uart2_ck, | |
| 550 | + &uart3_1510, | |
| 551 | + &uart3_16xx, | |
| 552 | + &usb_clk0, | |
| 553 | + &usb_hhc_ck1510, &usb_hhc_ck16xx, | |
| 554 | + &usb_dc_ck, | |
| 555 | + &mclk_1510, &mclk_16xx, &mclk_310, | |
| 556 | + &bclk_1510, &bclk_16xx, &bclk_310, | |
| 557 | + &mmc1_ck, | |
| 558 | + &mmc2_ck, | |
| 559 | + &cam_mclk, | |
| 560 | + &cam_exclk, | |
| 561 | + &cam_lclk, | |
| 562 | + &clk32k, | |
| 563 | + /* Virtual clocks */ | |
| 564 | + &i2c_fck, | |
| 565 | + &i2c_ick, | |
| 566 | + 0 | |
| 567 | +}; | |
| 568 | + | |
| 569 | +void omap_clk_adduser(struct clk *clk, qemu_irq user) | |
| 570 | +{ | |
| 571 | + qemu_irq *i; | |
| 572 | + | |
| 573 | + for (i = clk->users; *i; i ++); | |
| 574 | + *i = user; | |
| 575 | +} | |
| 576 | + | |
| 577 | +/* If a clock is allowed to idle, it is disabled automatically when | |
| 578 | + * all of clock domains using it are disabled. */ | |
| 579 | +int omap_clk_is_idle(struct clk *clk) | |
| 580 | +{ | |
| 581 | + struct clk *chld; | |
| 582 | + | |
| 583 | + if (!clk->enabled && (!clk->usecount || !(clk->flags && ALWAYS_ENABLED))) | |
| 584 | + return 1; | |
| 585 | + if (clk->usecount) | |
| 586 | + return 0; | |
| 587 | + | |
| 588 | + for (chld = clk->child1; chld; chld = chld->sibling) | |
| 589 | + if (!omap_clk_is_idle(chld)) | |
| 590 | + return 0; | |
| 591 | + return 1; | |
| 592 | +} | |
| 593 | + | |
| 594 | +struct clk *omap_findclk(struct omap_mpu_state_s *mpu, const char *name) | |
| 595 | +{ | |
| 596 | + struct clk *i; | |
| 597 | + | |
| 598 | + for (i = mpu->clks; i->name; i ++) | |
| 599 | + if (!strcmp(i->name, name) || (i->alias && !strcmp(i->alias, name))) | |
| 600 | + return i; | |
| 601 | + cpu_abort(mpu->env, "%s: %s not found\n", __FUNCTION__, name); | |
| 602 | +} | |
| 603 | + | |
| 604 | +void omap_clk_get(struct clk *clk) | |
| 605 | +{ | |
| 606 | + clk->usecount ++; | |
| 607 | +} | |
| 608 | + | |
| 609 | +void omap_clk_put(struct clk *clk) | |
| 610 | +{ | |
| 611 | + if (!(clk->usecount --)) | |
| 612 | + cpu_abort(cpu_single_env, "%s: %s is not in use\n", | |
| 613 | + __FUNCTION__, clk->name); | |
| 614 | +} | |
| 615 | + | |
| 616 | +static void omap_clk_update(struct clk *clk) | |
| 617 | +{ | |
| 618 | + int parent, running; | |
| 619 | + qemu_irq *user; | |
| 620 | + struct clk *i; | |
| 621 | + | |
| 622 | + if (clk->parent) | |
| 623 | + parent = clk->parent->running; | |
| 624 | + else | |
| 625 | + parent = 1; | |
| 626 | + | |
| 627 | + running = parent && (clk->enabled || | |
| 628 | + ((clk->flags & ALWAYS_ENABLED) && clk->usecount)); | |
| 629 | + if (clk->running != running) { | |
| 630 | + clk->running = running; | |
| 631 | + for (user = clk->users; *user; user ++) | |
| 632 | + qemu_set_irq(*user, running); | |
| 633 | + for (i = clk->child1; i; i = i->sibling) | |
| 634 | + omap_clk_update(i); | |
| 635 | + } | |
| 636 | +} | |
| 637 | + | |
| 638 | +static void omap_clk_rate_update_full(struct clk *clk, unsigned long int rate, | |
| 639 | + unsigned long int div, unsigned long int mult) | |
| 640 | +{ | |
| 641 | + struct clk *i; | |
| 642 | + qemu_irq *user; | |
| 643 | + | |
| 644 | + clk->rate = muldiv64(rate, mult, div); | |
| 645 | + if (clk->running) | |
| 646 | + for (user = clk->users; *user; user ++) | |
| 647 | + qemu_irq_raise(*user); | |
| 648 | + for (i = clk->child1; i; i = i->sibling) | |
| 649 | + omap_clk_rate_update_full(i, rate, | |
| 650 | + div * i->divisor, mult * i->multiplier); | |
| 651 | +} | |
| 652 | + | |
| 653 | +static void omap_clk_rate_update(struct clk *clk) | |
| 654 | +{ | |
| 655 | + struct clk *i; | |
| 656 | + unsigned long int div, mult = div = 1; | |
| 657 | + | |
| 658 | + for (i = clk; i->parent; i = i->parent) { | |
| 659 | + div *= i->divisor; | |
| 660 | + mult *= i->multiplier; | |
| 661 | + } | |
| 662 | + | |
| 663 | + omap_clk_rate_update_full(clk, i->rate, div, mult); | |
| 664 | +} | |
| 665 | + | |
| 666 | +void omap_clk_reparent(struct clk *clk, struct clk *parent) | |
| 667 | +{ | |
| 668 | + struct clk **p; | |
| 669 | + | |
| 670 | + if (clk->parent) { | |
| 671 | + for (p = &clk->parent->child1; *p != clk; p = &(*p)->sibling); | |
| 672 | + *p = clk->sibling; | |
| 673 | + } | |
| 674 | + | |
| 675 | + clk->parent = parent; | |
| 676 | + if (parent) { | |
| 677 | + clk->sibling = parent->child1; | |
| 678 | + parent->child1 = clk; | |
| 679 | + omap_clk_update(clk); | |
| 680 | + omap_clk_rate_update(clk); | |
| 681 | + } else | |
| 682 | + clk->sibling = 0; | |
| 683 | +} | |
| 684 | + | |
| 685 | +void omap_clk_onoff(struct clk *clk, int on) | |
| 686 | +{ | |
| 687 | + clk->enabled = on; | |
| 688 | + omap_clk_update(clk); | |
| 689 | +} | |
| 690 | + | |
| 691 | +void omap_clk_canidle(struct clk *clk, int can) | |
| 692 | +{ | |
| 693 | + if (can) | |
| 694 | + omap_clk_put(clk); | |
| 695 | + else | |
| 696 | + omap_clk_get(clk); | |
| 697 | +} | |
| 698 | + | |
| 699 | +void omap_clk_setrate(struct clk *clk, int divide, int multiply) | |
| 700 | +{ | |
| 701 | + clk->divisor = divide; | |
| 702 | + clk->multiplier = multiply; | |
| 703 | + omap_clk_rate_update(clk); | |
| 704 | +} | |
| 705 | + | |
| 706 | +int64_t omap_clk_getrate(omap_clk clk) | |
| 707 | +{ | |
| 708 | + return clk->rate; | |
| 709 | +} | |
| 710 | + | |
| 711 | +void omap_clk_init(struct omap_mpu_state_s *mpu) | |
| 712 | +{ | |
| 713 | + struct clk **i, *j, *k; | |
| 714 | + int count; | |
| 715 | + int flag; | |
| 716 | + | |
| 717 | + if (cpu_is_omap310(mpu)) | |
| 718 | + flag = CLOCK_IN_OMAP310; | |
| 719 | + else if (cpu_is_omap1510(mpu)) | |
| 720 | + flag = CLOCK_IN_OMAP1510; | |
| 721 | + else | |
| 722 | + return; | |
| 723 | + | |
| 724 | + for (i = onchip_clks, count = 0; *i; i ++) | |
| 725 | + if ((*i)->flags & flag) | |
| 726 | + count ++; | |
| 727 | + mpu->clks = (struct clk *) qemu_mallocz(sizeof(struct clk) * (count + 1)); | |
| 728 | + for (i = onchip_clks, j = mpu->clks; *i; i ++) | |
| 729 | + if ((*i)->flags & flag) { | |
| 730 | + memcpy(j, *i, sizeof(struct clk)); | |
| 731 | + for (k = mpu->clks; k < j; k ++) | |
| 732 | + if (j->parent && !strcmp(j->parent->name, k->name)) { | |
| 733 | + j->parent = k; | |
| 734 | + j->sibling = k->child1; | |
| 735 | + k->child1 = j; | |
| 736 | + } else if (k->parent && !strcmp(k->parent->name, j->name)) { | |
| 737 | + k->parent = j; | |
| 738 | + k->sibling = j->child1; | |
| 739 | + j->child1 = k; | |
| 740 | + } | |
| 741 | + j->divisor = j->divisor ?: 1; | |
| 742 | + j->multiplier = j->multiplier ?: 1; | |
| 743 | + j ++; | |
| 744 | + } | |
| 745 | +} | ... | ... |
hw/omap_lcd_template.h
0 → 100644
| 1 | +/* | |
| 2 | + * QEMU OMAP LCD Emulator templates | |
| 3 | + * | |
| 4 | + * Copyright (c) 2006 Andrzej Zaborowski <balrog@zabor.org> | |
| 5 | + * | |
| 6 | + * Redistribution and use in source and binary forms, with or without | |
| 7 | + * modification, are permitted provided that the following conditions | |
| 8 | + * are met: | |
| 9 | + * | |
| 10 | + * 1. Redistributions of source code must retain the above copyright | |
| 11 | + * notice, this list of conditions and the following disclaimer. | |
| 12 | + * 2. Redistributions in binary form must reproduce the above copyright | |
| 13 | + * notice, this list of conditions and the following disclaimer in | |
| 14 | + * the documentation and/or other materials provided with the | |
| 15 | + * distribution. | |
| 16 | + * | |
| 17 | + * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' | |
| 18 | + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, | |
| 19 | + * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A | |
| 20 | + * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR | |
| 21 | + * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, | |
| 22 | + * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, | |
| 23 | + * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR | |
| 24 | + * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY | |
| 25 | + * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | |
| 26 | + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | |
| 27 | + * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |
| 28 | + */ | |
| 29 | + | |
| 30 | +#if DEPTH == 8 | |
| 31 | +# define BPP 1 | |
| 32 | +# define PIXEL_TYPE uint8_t | |
| 33 | +#elif DEPTH == 15 || DEPTH == 16 | |
| 34 | +# define BPP 2 | |
| 35 | +# define PIXEL_TYPE uint16_t | |
| 36 | +#elif DEPTH == 32 | |
| 37 | +# define BPP 4 | |
| 38 | +# define PIXEL_TYPE uint32_t | |
| 39 | +#else | |
| 40 | +# error unsupport depth | |
| 41 | +#endif | |
| 42 | + | |
| 43 | +/* | |
| 44 | + * 2-bit colour | |
| 45 | + */ | |
| 46 | +static void glue(draw_line2_, DEPTH)( | |
| 47 | + uint8_t *d, const uint8_t *s, int width, const uint16_t *pal) | |
| 48 | +{ | |
| 49 | + uint8_t v, r, g, b; | |
| 50 | + | |
| 51 | + do { | |
| 52 | + v = ldub_raw((void *) s); | |
| 53 | + r = (pal[v & 3] >> 4) & 0xf0; | |
| 54 | + g = pal[v & 3] & 0xf0; | |
| 55 | + b = (pal[v & 3] << 4) & 0xf0; | |
| 56 | + ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b); | |
| 57 | + d += BPP; | |
| 58 | + v >>= 2; | |
| 59 | + r = (pal[v & 3] >> 4) & 0xf0; | |
| 60 | + g = pal[v & 3] & 0xf0; | |
| 61 | + b = (pal[v & 3] << 4) & 0xf0; | |
| 62 | + ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b); | |
| 63 | + d += BPP; | |
| 64 | + v >>= 2; | |
| 65 | + r = (pal[v & 3] >> 4) & 0xf0; | |
| 66 | + g = pal[v & 3] & 0xf0; | |
| 67 | + b = (pal[v & 3] << 4) & 0xf0; | |
| 68 | + ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b); | |
| 69 | + d += BPP; | |
| 70 | + v >>= 2; | |
| 71 | + r = (pal[v & 3] >> 4) & 0xf0; | |
| 72 | + g = pal[v & 3] & 0xf0; | |
| 73 | + b = (pal[v & 3] << 4) & 0xf0; | |
| 74 | + ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b); | |
| 75 | + d += BPP; | |
| 76 | + s ++; | |
| 77 | + width -= 4; | |
| 78 | + } while (width > 0); | |
| 79 | +} | |
| 80 | + | |
| 81 | +/* | |
| 82 | + * 4-bit colour | |
| 83 | + */ | |
| 84 | +static void glue(draw_line4_, DEPTH)( | |
| 85 | + uint8_t *d, const uint8_t *s, int width, const uint16_t *pal) | |
| 86 | +{ | |
| 87 | + uint8_t v, r, g, b; | |
| 88 | + | |
| 89 | + do { | |
| 90 | + v = ldub_raw((void *) s); | |
| 91 | + r = (pal[v & 0xf] >> 4) & 0xf0; | |
| 92 | + g = pal[v & 0xf] & 0xf0; | |
| 93 | + b = (pal[v & 0xf] << 4) & 0xf0; | |
| 94 | + ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b); | |
| 95 | + d += BPP; | |
| 96 | + v >>= 4; | |
| 97 | + r = (pal[v & 0xf] >> 4) & 0xf0; | |
| 98 | + g = pal[v & 0xf] & 0xf0; | |
| 99 | + b = (pal[v & 0xf] << 4) & 0xf0; | |
| 100 | + ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b); | |
| 101 | + d += BPP; | |
| 102 | + s ++; | |
| 103 | + width -= 2; | |
| 104 | + } while (width > 0); | |
| 105 | +} | |
| 106 | + | |
| 107 | +/* | |
| 108 | + * 8-bit colour | |
| 109 | + */ | |
| 110 | +static void glue(draw_line8_, DEPTH)( | |
| 111 | + uint8_t *d, const uint8_t *s, int width, const uint16_t *pal) | |
| 112 | +{ | |
| 113 | + uint8_t v, r, g, b; | |
| 114 | + | |
| 115 | + do { | |
| 116 | + v = ldub_raw((void *) s); | |
| 117 | + r = (pal[v] >> 4) & 0xf0; | |
| 118 | + g = pal[v] & 0xf0; | |
| 119 | + b = (pal[v] << 4) & 0xf0; | |
| 120 | + ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b); | |
| 121 | + s ++; | |
| 122 | + d += BPP; | |
| 123 | + } while (-- width != 0); | |
| 124 | +} | |
| 125 | + | |
| 126 | +/* | |
| 127 | + * 12-bit colour | |
| 128 | + */ | |
| 129 | +static void glue(draw_line12_, DEPTH)( | |
| 130 | + uint8_t *d, const uint8_t *s, int width, const uint16_t *pal) | |
| 131 | +{ | |
| 132 | + uint16_t v; | |
| 133 | + uint8_t r, g, b; | |
| 134 | + | |
| 135 | + do { | |
| 136 | + v = lduw_raw((void *) s); | |
| 137 | + r = (v >> 4) & 0xf0; | |
| 138 | + g = v & 0xf0; | |
| 139 | + b = (v << 4) & 0xf0; | |
| 140 | + ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b); | |
| 141 | + s += 2; | |
| 142 | + d += BPP; | |
| 143 | + } while (-- width != 0); | |
| 144 | +} | |
| 145 | + | |
| 146 | +/* | |
| 147 | + * 16-bit colour | |
| 148 | + */ | |
| 149 | +static void glue(draw_line16_, DEPTH)( | |
| 150 | + uint8_t *d, const uint8_t *s, int width, const uint16_t *pal) | |
| 151 | +{ | |
| 152 | +#if DEPTH == 16 && defined(WORDS_BIGENDIAN) == defined(TARGET_WORDS_BIGENDIAN) | |
| 153 | + memcpy(d, s, width * 2); | |
| 154 | +#else | |
| 155 | + uint16_t v; | |
| 156 | + uint8_t r, g, b; | |
| 157 | + | |
| 158 | + do { | |
| 159 | + v = lduw_raw((void *) s); | |
| 160 | + r = (v >> 8) & 0xf8; | |
| 161 | + g = (v >> 3) & 0xfc; | |
| 162 | + b = (v << 3) & 0xf8; | |
| 163 | + ((PIXEL_TYPE *) d)[0] = glue(rgb_to_pixel, DEPTH)(r, g, b); | |
| 164 | + s += 2; | |
| 165 | + d += BPP; | |
| 166 | + } while (-- width != 0); | |
| 167 | +#endif | |
| 168 | +} | |
| 169 | + | |
| 170 | +#undef DEPTH | |
| 171 | +#undef BPP | |
| 172 | +#undef PIXEL_TYPE | ... | ... |
hw/omap_lcdc.c
0 → 100644
| 1 | +/* | |
| 2 | + * OMAP LCD controller. | |
| 3 | + * | |
| 4 | + * Copyright (C) 2006-2007 Andrzej Zaborowski <balrog@zabor.org> | |
| 5 | + * | |
| 6 | + * This program is free software; you can redistribute it and/or | |
| 7 | + * modify it under the terms of the GNU General Public License as | |
| 8 | + * published by the Free Software Foundation; either version 2 of | |
| 9 | + * the License, or (at your option) any later version. | |
| 10 | + * | |
| 11 | + * This program is distributed in the hope that it will be useful, | |
| 12 | + * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
| 13 | + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
| 14 | + * GNU General Public License for more details. | |
| 15 | + * | |
| 16 | + * You should have received a copy of the GNU General Public License | |
| 17 | + * along with this program; if not, write to the Free Software | |
| 18 | + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, | |
| 19 | + * MA 02111-1307 USA | |
| 20 | + */ | |
| 21 | +#include "vl.h" | |
| 22 | + | |
| 23 | +struct omap_lcd_panel_s { | |
| 24 | + target_phys_addr_t base; | |
| 25 | + qemu_irq irq; | |
| 26 | + DisplayState *state; | |
| 27 | + ram_addr_t imif_base; | |
| 28 | + ram_addr_t emiff_base; | |
| 29 | + | |
| 30 | + int plm; | |
| 31 | + int tft; | |
| 32 | + int mono; | |
| 33 | + int enable; | |
| 34 | + int width; | |
| 35 | + int height; | |
| 36 | + int interrupts; | |
| 37 | + uint32_t timing[3]; | |
| 38 | + uint32_t subpanel; | |
| 39 | + uint32_t ctrl; | |
| 40 | + | |
| 41 | + struct omap_dma_lcd_channel_s *dma; | |
| 42 | + uint16_t palette[256]; | |
| 43 | + int palette_done; | |
| 44 | + int frame_done; | |
| 45 | + int invalidate; | |
| 46 | + int sync_error; | |
| 47 | +}; | |
| 48 | + | |
| 49 | +static void omap_lcd_interrupts(struct omap_lcd_panel_s *s) | |
| 50 | +{ | |
| 51 | + if (s->frame_done && (s->interrupts & 1)) { | |
| 52 | + qemu_irq_raise(s->irq); | |
| 53 | + return; | |
| 54 | + } | |
| 55 | + | |
| 56 | + if (s->palette_done && (s->interrupts & 2)) { | |
| 57 | + qemu_irq_raise(s->irq); | |
| 58 | + return; | |
| 59 | + } | |
| 60 | + | |
| 61 | + if (s->sync_error) { | |
| 62 | + qemu_irq_raise(s->irq); | |
| 63 | + return; | |
| 64 | + } | |
| 65 | + | |
| 66 | + qemu_irq_lower(s->irq); | |
| 67 | +} | |
| 68 | + | |
| 69 | +#include "pixel_ops.h" | |
| 70 | + | |
| 71 | +typedef void draw_line_func( | |
| 72 | + uint8_t *d, const uint8_t *s, int width, const uint16_t *pal); | |
| 73 | + | |
| 74 | +#define DEPTH 8 | |
| 75 | +#include "omap_lcd_template.h" | |
| 76 | +#define DEPTH 15 | |
| 77 | +#include "omap_lcd_template.h" | |
| 78 | +#define DEPTH 16 | |
| 79 | +#include "omap_lcd_template.h" | |
| 80 | +#define DEPTH 32 | |
| 81 | +#include "omap_lcd_template.h" | |
| 82 | + | |
| 83 | +static draw_line_func *draw_line_table2[33] = { | |
| 84 | + [0 ... 32] = 0, | |
| 85 | + [8] = draw_line2_8, | |
| 86 | + [15] = draw_line2_15, | |
| 87 | + [16] = draw_line2_16, | |
| 88 | + [32] = draw_line2_32, | |
| 89 | +}, *draw_line_table4[33] = { | |
| 90 | + [0 ... 32] = 0, | |
| 91 | + [8] = draw_line4_8, | |
| 92 | + [15] = draw_line4_15, | |
| 93 | + [16] = draw_line4_16, | |
| 94 | + [32] = draw_line4_32, | |
| 95 | +}, *draw_line_table8[33] = { | |
| 96 | + [0 ... 32] = 0, | |
| 97 | + [8] = draw_line8_8, | |
| 98 | + [15] = draw_line8_15, | |
| 99 | + [16] = draw_line8_16, | |
| 100 | + [32] = draw_line8_32, | |
| 101 | +}, *draw_line_table12[33] = { | |
| 102 | + [0 ... 32] = 0, | |
| 103 | + [8] = draw_line12_8, | |
| 104 | + [15] = draw_line12_15, | |
| 105 | + [16] = draw_line12_16, | |
| 106 | + [32] = draw_line12_32, | |
| 107 | +}, *draw_line_table16[33] = { | |
| 108 | + [0 ... 32] = 0, | |
| 109 | + [8] = draw_line16_8, | |
| 110 | + [15] = draw_line16_15, | |
| 111 | + [16] = draw_line16_16, | |
| 112 | + [32] = draw_line16_32, | |
| 113 | +}; | |
| 114 | + | |
| 115 | +void omap_update_display(void *opaque) | |
| 116 | +{ | |
| 117 | + struct omap_lcd_panel_s *omap_lcd = (struct omap_lcd_panel_s *) opaque; | |
| 118 | + draw_line_func *draw_line; | |
| 119 | + int size, dirty[2], minline, maxline, height; | |
| 120 | + int line, width, linesize, step, bpp, frame_offset; | |
| 121 | + ram_addr_t frame_base, scanline, newline, x; | |
| 122 | + uint8_t *s, *d; | |
| 123 | + | |
| 124 | + if (!omap_lcd || omap_lcd->plm == 1 || | |
| 125 | + !omap_lcd->enable || !omap_lcd->state->depth) | |
| 126 | + return; | |
| 127 | + | |
| 128 | + frame_offset = 0; | |
| 129 | + if (omap_lcd->plm != 2) { | |
| 130 | + memcpy(omap_lcd->palette, phys_ram_base + | |
| 131 | + omap_lcd->dma->phys_framebuffer[ | |
| 132 | + omap_lcd->dma->current_frame], 0x200); | |
| 133 | + switch (omap_lcd->palette[0] >> 12 & 7) { | |
| 134 | + case 3 ... 7: | |
| 135 | + frame_offset += 0x200; | |
| 136 | + break; | |
| 137 | + default: | |
| 138 | + frame_offset += 0x20; | |
| 139 | + } | |
| 140 | + } | |
| 141 | + | |
| 142 | + /* Colour depth */ | |
| 143 | + switch ((omap_lcd->palette[0] >> 12) & 7) { | |
| 144 | + case 1: | |
| 145 | + draw_line = draw_line_table2[omap_lcd->state->depth]; | |
| 146 | + bpp = 2; | |
| 147 | + break; | |
| 148 | + | |
| 149 | + case 2: | |
| 150 | + draw_line = draw_line_table4[omap_lcd->state->depth]; | |
| 151 | + bpp = 4; | |
| 152 | + break; | |
| 153 | + | |
| 154 | + case 3: | |
| 155 | + draw_line = draw_line_table8[omap_lcd->state->depth]; | |
| 156 | + bpp = 8; | |
| 157 | + break; | |
| 158 | + | |
| 159 | + case 4 ... 7: | |
| 160 | + if (!omap_lcd->tft) | |
| 161 | + draw_line = draw_line_table12[omap_lcd->state->depth]; | |
| 162 | + else | |
| 163 | + draw_line = draw_line_table16[omap_lcd->state->depth]; | |
| 164 | + bpp = 16; | |
| 165 | + break; | |
| 166 | + | |
| 167 | + default: | |
| 168 | + /* Unsupported at the moment. */ | |
| 169 | + return; | |
| 170 | + } | |
| 171 | + | |
| 172 | + /* Resolution */ | |
| 173 | + width = omap_lcd->width; | |
| 174 | + if (width != omap_lcd->state->width || | |
| 175 | + omap_lcd->height != omap_lcd->state->height) { | |
| 176 | + dpy_resize(omap_lcd->state, | |
| 177 | + omap_lcd->width, omap_lcd->height); | |
| 178 | + omap_lcd->invalidate = 1; | |
| 179 | + } | |
| 180 | + | |
| 181 | + if (omap_lcd->dma->current_frame == 0) | |
| 182 | + size = omap_lcd->dma->src_f1_bottom - omap_lcd->dma->src_f1_top; | |
| 183 | + else | |
| 184 | + size = omap_lcd->dma->src_f2_bottom - omap_lcd->dma->src_f2_top; | |
| 185 | + | |
| 186 | + if (frame_offset + ((width * omap_lcd->height * bpp) >> 3) > size + 2) { | |
| 187 | + omap_lcd->sync_error = 1; | |
| 188 | + omap_lcd_interrupts(omap_lcd); | |
| 189 | + omap_lcd->enable = 0; | |
| 190 | + return; | |
| 191 | + } | |
| 192 | + | |
| 193 | + /* Content */ | |
| 194 | + frame_base = omap_lcd->dma->phys_framebuffer[ | |
| 195 | + omap_lcd->dma->current_frame] + frame_offset; | |
| 196 | + omap_lcd->dma->condition |= 1 << omap_lcd->dma->current_frame; | |
| 197 | + if (omap_lcd->dma->interrupts & 1) | |
| 198 | + qemu_irq_raise(omap_lcd->dma->irq); | |
| 199 | + if (omap_lcd->dma->dual) | |
| 200 | + omap_lcd->dma->current_frame ^= 1; | |
| 201 | + | |
| 202 | + if (!omap_lcd->state->depth) | |
| 203 | + return; | |
| 204 | + | |
| 205 | + line = 0; | |
| 206 | + height = omap_lcd->height; | |
| 207 | + if (omap_lcd->subpanel & (1 << 31)) { | |
| 208 | + if (omap_lcd->subpanel & (1 << 29)) | |
| 209 | + line = (omap_lcd->subpanel >> 16) & 0x3ff; | |
| 210 | + else | |
| 211 | + height = (omap_lcd->subpanel >> 16) & 0x3ff; | |
| 212 | + /* TODO: fill the rest of the panel with DPD */ | |
| 213 | + } | |
| 214 | + step = width * bpp >> 3; | |
| 215 | + scanline = frame_base + step * line; | |
| 216 | + s = (uint8_t *) (phys_ram_base + scanline); | |
| 217 | + d = omap_lcd->state->data; | |
| 218 | + linesize = omap_lcd->state->linesize; | |
| 219 | + | |
| 220 | + dirty[0] = dirty[1] = | |
| 221 | + cpu_physical_memory_get_dirty(scanline, VGA_DIRTY_FLAG); | |
| 222 | + minline = height; | |
| 223 | + maxline = line; | |
| 224 | + for (; line < height; line ++) { | |
| 225 | + newline = scanline + step; | |
| 226 | + for (x = scanline + TARGET_PAGE_SIZE; x < newline; | |
| 227 | + x += TARGET_PAGE_SIZE) { | |
| 228 | + dirty[1] = cpu_physical_memory_get_dirty(x, VGA_DIRTY_FLAG); | |
| 229 | + dirty[0] |= dirty[1]; | |
| 230 | + } | |
| 231 | + if (dirty[0] || omap_lcd->invalidate) { | |
| 232 | + draw_line(d, s, width, omap_lcd->palette); | |
| 233 | + if (line < minline) | |
| 234 | + minline = line; | |
| 235 | + maxline = line + 1; | |
| 236 | + } | |
| 237 | + scanline = newline; | |
| 238 | + dirty[0] = dirty[1]; | |
| 239 | + s += step; | |
| 240 | + d += linesize; | |
| 241 | + } | |
| 242 | + | |
| 243 | + if (maxline >= minline) { | |
| 244 | + dpy_update(omap_lcd->state, 0, minline, width, maxline); | |
| 245 | + cpu_physical_memory_reset_dirty(frame_base + step * minline, | |
| 246 | + frame_base + step * maxline, VGA_DIRTY_FLAG); | |
| 247 | + } | |
| 248 | +} | |
| 249 | + | |
| 250 | +static int ppm_save(const char *filename, uint8_t *data, | |
| 251 | + int w, int h, int linesize) | |
| 252 | +{ | |
| 253 | + FILE *f; | |
| 254 | + uint8_t *d, *d1; | |
| 255 | + unsigned int v; | |
| 256 | + int y, x, bpp; | |
| 257 | + | |
| 258 | + f = fopen(filename, "wb"); | |
| 259 | + if (!f) | |
| 260 | + return -1; | |
| 261 | + fprintf(f, "P6\n%d %d\n%d\n", w, h, 255); | |
| 262 | + d1 = data; | |
| 263 | + bpp = linesize / w; | |
| 264 | + for (y = 0; y < h; y ++) { | |
| 265 | + d = d1; | |
| 266 | + for (x = 0; x < w; x ++) { | |
| 267 | + v = *(uint32_t *) d; | |
| 268 | + switch (bpp) { | |
| 269 | + case 2: | |
| 270 | + fputc((v >> 8) & 0xf8, f); | |
| 271 | + fputc((v >> 3) & 0xfc, f); | |
| 272 | + fputc((v << 3) & 0xf8, f); | |
| 273 | + break; | |
| 274 | + case 3: | |
| 275 | + case 4: | |
| 276 | + default: | |
| 277 | + fputc((v >> 16) & 0xff, f); | |
| 278 | + fputc((v >> 8) & 0xff, f); | |
| 279 | + fputc((v) & 0xff, f); | |
| 280 | + break; | |
| 281 | + } | |
| 282 | + d += bpp; | |
| 283 | + } | |
| 284 | + d1 += linesize; | |
| 285 | + } | |
| 286 | + fclose(f); | |
| 287 | + return 0; | |
| 288 | +} | |
| 289 | + | |
| 290 | +void omap_screen_dump(void *opaque, const char *filename) { | |
| 291 | + struct omap_lcd_panel_s *omap_lcd = opaque; | |
| 292 | + omap_update_display(opaque); | |
| 293 | + if (omap_lcd && omap_lcd->state->data) | |
| 294 | + ppm_save(filename, omap_lcd->state->data, | |
| 295 | + omap_lcd->width, omap_lcd->height, | |
| 296 | + omap_lcd->state->linesize); | |
| 297 | +} | |
| 298 | + | |
| 299 | +void omap_invalidate_display(void *opaque) { | |
| 300 | + struct omap_lcd_panel_s *omap_lcd = opaque; | |
| 301 | + omap_lcd->invalidate = 1; | |
| 302 | +} | |
| 303 | + | |
| 304 | +void omap_lcd_update(struct omap_lcd_panel_s *s) { | |
| 305 | + if (!s->enable) { | |
| 306 | + s->dma->current_frame = -1; | |
| 307 | + s->sync_error = 0; | |
| 308 | + if (s->plm != 1) | |
| 309 | + s->frame_done = 1; | |
| 310 | + omap_lcd_interrupts(s); | |
| 311 | + return; | |
| 312 | + } | |
| 313 | + | |
| 314 | + if (s->dma->current_frame == -1) { | |
| 315 | + s->frame_done = 0; | |
| 316 | + s->palette_done = 0; | |
| 317 | + s->dma->current_frame = 0; | |
| 318 | + } | |
| 319 | + | |
| 320 | + if (!s->dma->mpu->port[s->dma->src].addr_valid(s->dma->mpu, | |
| 321 | + s->dma->src_f1_top) || | |
| 322 | + !s->dma->mpu->port[ | |
| 323 | + s->dma->src].addr_valid(s->dma->mpu, | |
| 324 | + s->dma->src_f1_bottom) || | |
| 325 | + (s->dma->dual && | |
| 326 | + (!s->dma->mpu->port[ | |
| 327 | + s->dma->src].addr_valid(s->dma->mpu, | |
| 328 | + s->dma->src_f2_top) || | |
| 329 | + !s->dma->mpu->port[ | |
| 330 | + s->dma->src].addr_valid(s->dma->mpu, | |
| 331 | + s->dma->src_f2_bottom)))) { | |
| 332 | + s->dma->condition |= 1 << 2; | |
| 333 | + if (s->dma->interrupts & (1 << 1)) | |
| 334 | + qemu_irq_raise(s->dma->irq); | |
| 335 | + s->enable = 0; | |
| 336 | + return; | |
| 337 | + } | |
| 338 | + | |
| 339 | + if (s->dma->src == imif) { | |
| 340 | + /* Framebuffers are in SRAM */ | |
| 341 | + s->dma->phys_framebuffer[0] = s->imif_base + | |
| 342 | + s->dma->src_f1_top - OMAP_IMIF_BASE; | |
| 343 | + | |
| 344 | + s->dma->phys_framebuffer[1] = s->imif_base + | |
| 345 | + s->dma->src_f2_top - OMAP_IMIF_BASE; | |
| 346 | + } else { | |
| 347 | + /* Framebuffers are in RAM */ | |
| 348 | + s->dma->phys_framebuffer[0] = s->emiff_base + | |
| 349 | + s->dma->src_f1_top - OMAP_EMIFF_BASE; | |
| 350 | + | |
| 351 | + s->dma->phys_framebuffer[1] = s->emiff_base + | |
| 352 | + s->dma->src_f2_top - OMAP_EMIFF_BASE; | |
| 353 | + } | |
| 354 | + | |
| 355 | + if (s->plm != 2 && !s->palette_done) { | |
| 356 | + memcpy(s->palette, phys_ram_base + | |
| 357 | + s->dma->phys_framebuffer[s->dma->current_frame], 0x200); | |
| 358 | + s->palette_done = 1; | |
| 359 | + omap_lcd_interrupts(s); | |
| 360 | + } | |
| 361 | +} | |
| 362 | + | |
| 363 | +static uint32_t omap_lcdc_read(void *opaque, target_phys_addr_t addr) | |
| 364 | +{ | |
| 365 | + struct omap_lcd_panel_s *s = (struct omap_lcd_panel_s *) opaque; | |
| 366 | + int offset = addr - s->base; | |
| 367 | + | |
| 368 | + switch (offset) { | |
| 369 | + case 0x00: /* LCD_CONTROL */ | |
| 370 | + return (s->tft << 23) | (s->plm << 20) | | |
| 371 | + (s->tft << 7) | (s->interrupts << 3) | | |
| 372 | + (s->mono << 1) | s->enable | s->ctrl | 0xfe000c34; | |
| 373 | + | |
| 374 | + case 0x04: /* LCD_TIMING0 */ | |
| 375 | + return (s->timing[0] << 10) | (s->width - 1) | 0x0000000f; | |
| 376 | + | |
| 377 | + case 0x08: /* LCD_TIMING1 */ | |
| 378 | + return (s->timing[1] << 10) | (s->height - 1); | |
| 379 | + | |
| 380 | + case 0x0c: /* LCD_TIMING2 */ | |
| 381 | + return s->timing[2] | 0xfc000000; | |
| 382 | + | |
| 383 | + case 0x10: /* LCD_STATUS */ | |
| 384 | + return (s->palette_done << 6) | (s->sync_error << 2) | s->frame_done; | |
| 385 | + | |
| 386 | + case 0x14: /* LCD_SUBPANEL */ | |
| 387 | + return s->subpanel; | |
| 388 | + | |
| 389 | + default: | |
| 390 | + break; | |
| 391 | + } | |
| 392 | + OMAP_BAD_REG(addr); | |
| 393 | + return 0; | |
| 394 | +} | |
| 395 | + | |
| 396 | +static void omap_lcdc_write(void *opaque, target_phys_addr_t addr, | |
| 397 | + uint32_t value) | |
| 398 | +{ | |
| 399 | + struct omap_lcd_panel_s *s = (struct omap_lcd_panel_s *) opaque; | |
| 400 | + int offset = addr - s->base; | |
| 401 | + | |
| 402 | + switch (offset) { | |
| 403 | + case 0x00: /* LCD_CONTROL */ | |
| 404 | + s->plm = (value >> 20) & 3; | |
| 405 | + s->tft = (value >> 7) & 1; | |
| 406 | + s->interrupts = (value >> 3) & 3; | |
| 407 | + s->mono = (value >> 1) & 1; | |
| 408 | + s->ctrl = value & 0x01cff300; | |
| 409 | + if (s->enable != (value & 1)) { | |
| 410 | + s->enable = value & 1; | |
| 411 | + omap_lcd_update(s); | |
| 412 | + } | |
| 413 | + break; | |
| 414 | + | |
| 415 | + case 0x04: /* LCD_TIMING0 */ | |
| 416 | + s->timing[0] = value >> 10; | |
| 417 | + s->width = (value & 0x3ff) + 1; | |
| 418 | + break; | |
| 419 | + | |
| 420 | + case 0x08: /* LCD_TIMING1 */ | |
| 421 | + s->timing[1] = value >> 10; | |
| 422 | + s->height = (value & 0x3ff) + 1; | |
| 423 | + break; | |
| 424 | + | |
| 425 | + case 0x0c: /* LCD_TIMING2 */ | |
| 426 | + s->timing[2] = value; | |
| 427 | + break; | |
| 428 | + | |
| 429 | + case 0x10: /* LCD_STATUS */ | |
| 430 | + break; | |
| 431 | + | |
| 432 | + case 0x14: /* LCD_SUBPANEL */ | |
| 433 | + s->subpanel = value & 0xa1ffffff; | |
| 434 | + break; | |
| 435 | + | |
| 436 | + default: | |
| 437 | + OMAP_BAD_REG(addr); | |
| 438 | + } | |
| 439 | +} | |
| 440 | + | |
| 441 | +static CPUReadMemoryFunc *omap_lcdc_readfn[] = { | |
| 442 | + omap_lcdc_read, | |
| 443 | + omap_lcdc_read, | |
| 444 | + omap_lcdc_read, | |
| 445 | +}; | |
| 446 | + | |
| 447 | +static CPUWriteMemoryFunc *omap_lcdc_writefn[] = { | |
| 448 | + omap_lcdc_write, | |
| 449 | + omap_lcdc_write, | |
| 450 | + omap_lcdc_write, | |
| 451 | +}; | |
| 452 | + | |
| 453 | +void omap_lcdc_reset(struct omap_lcd_panel_s *s) | |
| 454 | +{ | |
| 455 | + s->dma->current_frame = -1; | |
| 456 | + s->plm = 0; | |
| 457 | + s->tft = 0; | |
| 458 | + s->mono = 0; | |
| 459 | + s->enable = 0; | |
| 460 | + s->width = 0; | |
| 461 | + s->height = 0; | |
| 462 | + s->interrupts = 0; | |
| 463 | + s->timing[0] = 0; | |
| 464 | + s->timing[1] = 0; | |
| 465 | + s->timing[2] = 0; | |
| 466 | + s->subpanel = 0; | |
| 467 | + s->palette_done = 0; | |
| 468 | + s->frame_done = 0; | |
| 469 | + s->sync_error = 0; | |
| 470 | + s->invalidate = 1; | |
| 471 | + s->subpanel = 0; | |
| 472 | + s->ctrl = 0; | |
| 473 | +} | |
| 474 | + | |
| 475 | +struct omap_lcd_panel_s *omap_lcdc_init(target_phys_addr_t base, qemu_irq irq, | |
| 476 | + struct omap_dma_lcd_channel_s *dma, DisplayState *ds, | |
| 477 | + ram_addr_t imif_base, ram_addr_t emiff_base, omap_clk clk) | |
| 478 | +{ | |
| 479 | + int iomemtype; | |
| 480 | + struct omap_lcd_panel_s *s = (struct omap_lcd_panel_s *) | |
| 481 | + qemu_mallocz(sizeof(struct omap_lcd_panel_s)); | |
| 482 | + | |
| 483 | + s->irq = irq; | |
| 484 | + s->dma = dma; | |
| 485 | + s->base = base; | |
| 486 | + s->state = ds; | |
| 487 | + s->imif_base = imif_base; | |
| 488 | + s->emiff_base = emiff_base; | |
| 489 | + omap_lcdc_reset(s); | |
| 490 | + | |
| 491 | + iomemtype = cpu_register_io_memory(0, omap_lcdc_readfn, | |
| 492 | + omap_lcdc_writefn, s); | |
| 493 | + cpu_register_physical_memory(s->base, 0x100, iomemtype); | |
| 494 | + | |
| 495 | + graphic_console_init(ds, omap_update_display, | |
| 496 | + omap_invalidate_display, omap_screen_dump, s); | |
| 497 | + | |
| 498 | + return s; | |
| 499 | +} | ... | ... |