Commit c3d2689d88159291ef8af338b5f01cfbe5551d2c

Authored by balrog
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

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 +}
... ...