musicpal.c 44.5 KB
Newer Older
1 2 3 4 5 6 7 8
/*
 * Marvell MV88W8618 / Freecom MusicPal emulation.
 *
 * Copyright (c) 2008 Jan Kiszka
 *
 * This code is licenced under the GNU GPL v2.
 */

P
Paul Brook 已提交
9
#include "sysbus.h"
10 11 12 13 14 15 16 17 18 19 20 21
#include "arm-misc.h"
#include "devices.h"
#include "net.h"
#include "sysemu.h"
#include "boards.h"
#include "pc.h"
#include "qemu-timer.h"
#include "block.h"
#include "flash.h"
#include "console.h"
#include "i2c.h"

22 23 24
#define MP_MISC_BASE            0x80002000
#define MP_MISC_SIZE            0x00001000

25 26 27
#define MP_ETH_BASE             0x80008000
#define MP_ETH_SIZE             0x00001000

28 29 30
#define MP_WLAN_BASE            0x8000C000
#define MP_WLAN_SIZE            0x00000800

31 32 33
#define MP_UART1_BASE           0x8000C840
#define MP_UART2_BASE           0x8000C940

34 35 36
#define MP_GPIO_BASE            0x8000D000
#define MP_GPIO_SIZE            0x00001000

37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57
#define MP_FLASHCFG_BASE        0x90006000
#define MP_FLASHCFG_SIZE        0x00001000

#define MP_AUDIO_BASE           0x90007000

#define MP_PIC_BASE             0x90008000
#define MP_PIC_SIZE             0x00001000

#define MP_PIT_BASE             0x90009000
#define MP_PIT_SIZE             0x00001000

#define MP_LCD_BASE             0x9000c000
#define MP_LCD_SIZE             0x00001000

#define MP_SRAM_BASE            0xC0000000
#define MP_SRAM_SIZE            0x00020000

#define MP_RAM_DEFAULT_SIZE     32*1024*1024
#define MP_FLASH_SIZE_MAX       32*1024*1024

#define MP_TIMER1_IRQ           4
P
Paul Brook 已提交
58 59
#define MP_TIMER2_IRQ           5
#define MP_TIMER3_IRQ           6
60 61 62 63 64 65 66 67 68 69
#define MP_TIMER4_IRQ           7
#define MP_EHCI_IRQ             8
#define MP_ETH_IRQ              9
#define MP_UART1_IRQ            11
#define MP_UART2_IRQ            11
#define MP_GPIO_IRQ             12
#define MP_RTC_IRQ              28
#define MP_AUDIO_IRQ            30

/* Wolfson 8750 I2C address */
J
Jan Kiszka 已提交
70
#define MP_WM_ADDR              0x1A
71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142

/* Ethernet register offsets */
#define MP_ETH_SMIR             0x010
#define MP_ETH_PCXR             0x408
#define MP_ETH_SDCMR            0x448
#define MP_ETH_ICR              0x450
#define MP_ETH_IMR              0x458
#define MP_ETH_FRDP0            0x480
#define MP_ETH_FRDP1            0x484
#define MP_ETH_FRDP2            0x488
#define MP_ETH_FRDP3            0x48C
#define MP_ETH_CRDP0            0x4A0
#define MP_ETH_CRDP1            0x4A4
#define MP_ETH_CRDP2            0x4A8
#define MP_ETH_CRDP3            0x4AC
#define MP_ETH_CTDP0            0x4E0
#define MP_ETH_CTDP1            0x4E4
#define MP_ETH_CTDP2            0x4E8
#define MP_ETH_CTDP3            0x4EC

/* MII PHY access */
#define MP_ETH_SMIR_DATA        0x0000FFFF
#define MP_ETH_SMIR_ADDR        0x03FF0000
#define MP_ETH_SMIR_OPCODE      (1 << 26) /* Read value */
#define MP_ETH_SMIR_RDVALID     (1 << 27)

/* PHY registers */
#define MP_ETH_PHY1_BMSR        0x00210000
#define MP_ETH_PHY1_PHYSID1     0x00410000
#define MP_ETH_PHY1_PHYSID2     0x00610000

#define MP_PHY_BMSR_LINK        0x0004
#define MP_PHY_BMSR_AUTONEG     0x0008

#define MP_PHY_88E3015          0x01410E20

/* TX descriptor status */
#define MP_ETH_TX_OWN           (1 << 31)

/* RX descriptor status */
#define MP_ETH_RX_OWN           (1 << 31)

/* Interrupt cause/mask bits */
#define MP_ETH_IRQ_RX_BIT       0
#define MP_ETH_IRQ_RX           (1 << MP_ETH_IRQ_RX_BIT)
#define MP_ETH_IRQ_TXHI_BIT     2
#define MP_ETH_IRQ_TXLO_BIT     3

/* Port config bits */
#define MP_ETH_PCXR_2BSM_BIT    28 /* 2-byte incoming suffix */

/* SDMA command bits */
#define MP_ETH_CMD_TXHI         (1 << 23)
#define MP_ETH_CMD_TXLO         (1 << 22)

typedef struct mv88w8618_tx_desc {
    uint32_t cmdstat;
    uint16_t res;
    uint16_t bytes;
    uint32_t buffer;
    uint32_t next;
} mv88w8618_tx_desc;

typedef struct mv88w8618_rx_desc {
    uint32_t cmdstat;
    uint16_t bytes;
    uint16_t buffer_size;
    uint32_t buffer;
    uint32_t next;
} mv88w8618_rx_desc;

typedef struct mv88w8618_eth_state {
P
Paul Brook 已提交
143
    SysBusDevice busdev;
144 145 146 147
    qemu_irq irq;
    uint32_t smir;
    uint32_t icr;
    uint32_t imr;
148
    int mmio_index;
J
Jan Kiszka 已提交
149
    uint32_t vlan_header;
P
pbrook 已提交
150 151 152 153
    uint32_t tx_queue[2];
    uint32_t rx_queue[4];
    uint32_t frx_queue[4];
    uint32_t cur_rx[4];
154
    NICState *nic;
155
    NICConf conf;
156 157
} mv88w8618_eth_state;

P
pbrook 已提交
158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177
static void eth_rx_desc_put(uint32_t addr, mv88w8618_rx_desc *desc)
{
    cpu_to_le32s(&desc->cmdstat);
    cpu_to_le16s(&desc->bytes);
    cpu_to_le16s(&desc->buffer_size);
    cpu_to_le32s(&desc->buffer);
    cpu_to_le32s(&desc->next);
    cpu_physical_memory_write(addr, (void *)desc, sizeof(*desc));
}

static void eth_rx_desc_get(uint32_t addr, mv88w8618_rx_desc *desc)
{
    cpu_physical_memory_read(addr, (void *)desc, sizeof(*desc));
    le32_to_cpus(&desc->cmdstat);
    le16_to_cpus(&desc->bytes);
    le16_to_cpus(&desc->buffer_size);
    le32_to_cpus(&desc->buffer);
    le32_to_cpus(&desc->next);
}

178
static int eth_can_receive(VLANClientState *nc)
179 180 181 182
{
    return 1;
}

183
static ssize_t eth_receive(VLANClientState *nc, const uint8_t *buf, size_t size)
184
{
185
    mv88w8618_eth_state *s = DO_UPCAST(NICState, nc, nc)->opaque;
P
pbrook 已提交
186 187
    uint32_t desc_addr;
    mv88w8618_rx_desc desc;
188 189 190
    int i;

    for (i = 0; i < 4; i++) {
P
pbrook 已提交
191
        desc_addr = s->cur_rx[i];
J
Jan Kiszka 已提交
192
        if (!desc_addr) {
193
            continue;
J
Jan Kiszka 已提交
194
        }
195
        do {
P
pbrook 已提交
196 197 198 199 200 201 202
            eth_rx_desc_get(desc_addr, &desc);
            if ((desc.cmdstat & MP_ETH_RX_OWN) && desc.buffer_size >= size) {
                cpu_physical_memory_write(desc.buffer + s->vlan_header,
                                          buf, size);
                desc.bytes = size + s->vlan_header;
                desc.cmdstat &= ~MP_ETH_RX_OWN;
                s->cur_rx[i] = desc.next;
203 204

                s->icr |= MP_ETH_IRQ_RX;
J
Jan Kiszka 已提交
205
                if (s->icr & s->imr) {
206
                    qemu_irq_raise(s->irq);
J
Jan Kiszka 已提交
207
                }
P
pbrook 已提交
208
                eth_rx_desc_put(desc_addr, &desc);
209
                return size;
210
            }
P
pbrook 已提交
211 212
            desc_addr = desc.next;
        } while (desc_addr != s->rx_queue[i]);
213
    }
214
    return size;
215 216
}

P
pbrook 已提交
217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236
static void eth_tx_desc_put(uint32_t addr, mv88w8618_tx_desc *desc)
{
    cpu_to_le32s(&desc->cmdstat);
    cpu_to_le16s(&desc->res);
    cpu_to_le16s(&desc->bytes);
    cpu_to_le32s(&desc->buffer);
    cpu_to_le32s(&desc->next);
    cpu_physical_memory_write(addr, (void *)desc, sizeof(*desc));
}

static void eth_tx_desc_get(uint32_t addr, mv88w8618_tx_desc *desc)
{
    cpu_physical_memory_read(addr, (void *)desc, sizeof(*desc));
    le32_to_cpus(&desc->cmdstat);
    le16_to_cpus(&desc->res);
    le16_to_cpus(&desc->bytes);
    le32_to_cpus(&desc->buffer);
    le32_to_cpus(&desc->next);
}

237 238
static void eth_send(mv88w8618_eth_state *s, int queue_index)
{
P
pbrook 已提交
239 240
    uint32_t desc_addr = s->tx_queue[queue_index];
    mv88w8618_tx_desc desc;
241
    uint32_t next_desc;
P
pbrook 已提交
242 243 244
    uint8_t buf[2048];
    int len;

245
    do {
P
pbrook 已提交
246
        eth_tx_desc_get(desc_addr, &desc);
247
        next_desc = desc.next;
P
pbrook 已提交
248 249 250 251
        if (desc.cmdstat & MP_ETH_TX_OWN) {
            len = desc.bytes;
            if (len < 2048) {
                cpu_physical_memory_read(desc.buffer, buf, len);
252
                qemu_send_packet(&s->nic->nc, buf, len);
P
pbrook 已提交
253 254
            }
            desc.cmdstat &= ~MP_ETH_TX_OWN;
255
            s->icr |= 1 << (MP_ETH_IRQ_TXLO_BIT - queue_index);
P
pbrook 已提交
256
            eth_tx_desc_put(desc_addr, &desc);
257
        }
258
        desc_addr = next_desc;
P
pbrook 已提交
259
    } while (desc_addr != s->tx_queue[queue_index]);
260 261
}

A
Anthony Liguori 已提交
262
static uint32_t mv88w8618_eth_read(void *opaque, target_phys_addr_t offset)
263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289
{
    mv88w8618_eth_state *s = opaque;

    switch (offset) {
    case MP_ETH_SMIR:
        if (s->smir & MP_ETH_SMIR_OPCODE) {
            switch (s->smir & MP_ETH_SMIR_ADDR) {
            case MP_ETH_PHY1_BMSR:
                return MP_PHY_BMSR_LINK | MP_PHY_BMSR_AUTONEG |
                       MP_ETH_SMIR_RDVALID;
            case MP_ETH_PHY1_PHYSID1:
                return (MP_PHY_88E3015 >> 16) | MP_ETH_SMIR_RDVALID;
            case MP_ETH_PHY1_PHYSID2:
                return (MP_PHY_88E3015 & 0xFFFF) | MP_ETH_SMIR_RDVALID;
            default:
                return MP_ETH_SMIR_RDVALID;
            }
        }
        return 0;

    case MP_ETH_ICR:
        return s->icr;

    case MP_ETH_IMR:
        return s->imr;

    case MP_ETH_FRDP0 ... MP_ETH_FRDP3:
P
pbrook 已提交
290
        return s->frx_queue[(offset - MP_ETH_FRDP0)/4];
291 292

    case MP_ETH_CRDP0 ... MP_ETH_CRDP3:
P
pbrook 已提交
293
        return s->rx_queue[(offset - MP_ETH_CRDP0)/4];
294 295

    case MP_ETH_CTDP0 ... MP_ETH_CTDP3:
P
pbrook 已提交
296
        return s->tx_queue[(offset - MP_ETH_CTDP0)/4];
297 298 299 300 301 302

    default:
        return 0;
    }
}

A
Anthony Liguori 已提交
303
static void mv88w8618_eth_write(void *opaque, target_phys_addr_t offset,
304 305 306 307 308 309 310 311 312 313 314 315 316 317
                                uint32_t value)
{
    mv88w8618_eth_state *s = opaque;

    switch (offset) {
    case MP_ETH_SMIR:
        s->smir = value;
        break;

    case MP_ETH_PCXR:
        s->vlan_header = ((value >> MP_ETH_PCXR_2BSM_BIT) & 1) * 2;
        break;

    case MP_ETH_SDCMR:
J
Jan Kiszka 已提交
318
        if (value & MP_ETH_CMD_TXHI) {
319
            eth_send(s, 1);
J
Jan Kiszka 已提交
320 321
        }
        if (value & MP_ETH_CMD_TXLO) {
322
            eth_send(s, 0);
J
Jan Kiszka 已提交
323 324
        }
        if (value & (MP_ETH_CMD_TXHI | MP_ETH_CMD_TXLO) && s->icr & s->imr) {
325
            qemu_irq_raise(s->irq);
J
Jan Kiszka 已提交
326
        }
327 328 329 330 331 332 333 334
        break;

    case MP_ETH_ICR:
        s->icr &= value;
        break;

    case MP_ETH_IMR:
        s->imr = value;
J
Jan Kiszka 已提交
335
        if (s->icr & s->imr) {
336
            qemu_irq_raise(s->irq);
J
Jan Kiszka 已提交
337
        }
338 339 340
        break;

    case MP_ETH_FRDP0 ... MP_ETH_FRDP3:
P
pbrook 已提交
341
        s->frx_queue[(offset - MP_ETH_FRDP0)/4] = value;
342 343 344 345
        break;

    case MP_ETH_CRDP0 ... MP_ETH_CRDP3:
        s->rx_queue[(offset - MP_ETH_CRDP0)/4] =
P
pbrook 已提交
346
            s->cur_rx[(offset - MP_ETH_CRDP0)/4] = value;
347 348 349
        break;

    case MP_ETH_CTDP0 ... MP_ETH_CTDP3:
P
pbrook 已提交
350
        s->tx_queue[(offset - MP_ETH_CTDP0)/4] = value;
351 352 353 354
        break;
    }
}

355
static CPUReadMemoryFunc * const mv88w8618_eth_readfn[] = {
356 357 358 359 360
    mv88w8618_eth_read,
    mv88w8618_eth_read,
    mv88w8618_eth_read
};

361
static CPUWriteMemoryFunc * const mv88w8618_eth_writefn[] = {
362 363 364 365 366
    mv88w8618_eth_write,
    mv88w8618_eth_write,
    mv88w8618_eth_write
};

367
static void eth_cleanup(VLANClientState *nc)
368
{
369
    mv88w8618_eth_state *s = DO_UPCAST(NICState, nc, nc)->opaque;
370

371
    s->nic = NULL;
372 373
}

374 375 376 377 378 379 380 381
static NetClientInfo net_mv88w8618_info = {
    .type = NET_CLIENT_TYPE_NIC,
    .size = sizeof(NICState),
    .can_receive = eth_can_receive,
    .receive = eth_receive,
    .cleanup = eth_cleanup,
};

382
static int mv88w8618_eth_init(SysBusDevice *dev)
383
{
P
Paul Brook 已提交
384
    mv88w8618_eth_state *s = FROM_SYSBUS(mv88w8618_eth_state, dev);
385

P
Paul Brook 已提交
386
    sysbus_init_irq(dev, &s->irq);
387 388
    s->nic = qemu_new_nic(&net_mv88w8618_info, &s->conf,
                          dev->qdev.info->name, dev->qdev.id, s);
389
    s->mmio_index = cpu_register_io_memory(mv88w8618_eth_readfn,
390
                                           mv88w8618_eth_writefn, s);
P
Paul Brook 已提交
391
    sysbus_init_mmio(dev, MP_ETH_SIZE, s->mmio_index);
392
    return 0;
393 394
}

J
Jan Kiszka 已提交
395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417
static const VMStateDescription mv88w8618_eth_vmsd = {
    .name = "mv88w8618_eth",
    .version_id = 1,
    .minimum_version_id = 1,
    .minimum_version_id_old = 1,
    .fields = (VMStateField[]) {
        VMSTATE_UINT32(smir, mv88w8618_eth_state),
        VMSTATE_UINT32(icr, mv88w8618_eth_state),
        VMSTATE_UINT32(imr, mv88w8618_eth_state),
        VMSTATE_UINT32(vlan_header, mv88w8618_eth_state),
        VMSTATE_UINT32_ARRAY(tx_queue, mv88w8618_eth_state, 2),
        VMSTATE_UINT32_ARRAY(rx_queue, mv88w8618_eth_state, 4),
        VMSTATE_UINT32_ARRAY(frx_queue, mv88w8618_eth_state, 4),
        VMSTATE_UINT32_ARRAY(cur_rx, mv88w8618_eth_state, 4),
        VMSTATE_END_OF_LIST()
    }
};

static SysBusDeviceInfo mv88w8618_eth_info = {
    .init = mv88w8618_eth_init,
    .qdev.name = "mv88w8618_eth",
    .qdev.size = sizeof(mv88w8618_eth_state),
    .qdev.vmsd = &mv88w8618_eth_vmsd,
418 419 420 421
    .qdev.props = (Property[]) {
        DEFINE_NIC_PROPERTIES(mv88w8618_eth_state, conf),
        DEFINE_PROP_END_OF_LIST(),
    },
J
Jan Kiszka 已提交
422 423
};

424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443
/* LCD register offsets */
#define MP_LCD_IRQCTRL          0x180
#define MP_LCD_IRQSTAT          0x184
#define MP_LCD_SPICTRL          0x1ac
#define MP_LCD_INST             0x1bc
#define MP_LCD_DATA             0x1c0

/* Mode magics */
#define MP_LCD_SPI_DATA         0x00100011
#define MP_LCD_SPI_CMD          0x00104011
#define MP_LCD_SPI_INVALID      0x00000000

/* Commmands */
#define MP_LCD_INST_SETPAGE0    0xB0
/* ... */
#define MP_LCD_INST_SETPAGE7    0xB7

#define MP_LCD_TEXTCOLOR        0xe0e0ff /* RRGGBB */

typedef struct musicpal_lcd_state {
P
Paul Brook 已提交
444
    SysBusDevice busdev;
445
    uint32_t brightness;
446 447
    uint32_t mode;
    uint32_t irqctrl;
J
Jan Kiszka 已提交
448 449
    uint32_t page;
    uint32_t page_off;
450 451 452 453
    DisplayState *ds;
    uint8_t video_ram[128*64/8];
} musicpal_lcd_state;

454
static uint8_t scale_lcd_color(musicpal_lcd_state *s, uint8_t col)
455
{
456 457 458 459
    switch (s->brightness) {
    case 7:
        return col;
    case 0:
460 461
        return 0;
    default:
462
        return (col * s->brightness) / 7;
463 464 465
    }
}

466 467 468 469 470
#define SET_LCD_PIXEL(depth, type) \
static inline void glue(set_lcd_pixel, depth) \
        (musicpal_lcd_state *s, int x, int y, type col) \
{ \
    int dx, dy; \
471
    type *pixel = &((type *) ds_get_data(s->ds))[(y * 128 * 3 + x) * 3]; \
472 473 474 475
\
    for (dy = 0; dy < 3; dy++, pixel += 127 * 3) \
        for (dx = 0; dx < 3; dx++, pixel++) \
            *pixel = col; \
476
}
477 478 479 480 481
SET_LCD_PIXEL(8, uint8_t)
SET_LCD_PIXEL(16, uint16_t)
SET_LCD_PIXEL(32, uint32_t)

#include "pixel_ops.h"
482 483 484 485

static void lcd_refresh(void *opaque)
{
    musicpal_lcd_state *s = opaque;
486
    int x, y, col;
487

488
    switch (ds_get_bits_per_pixel(s->ds)) {
489 490 491 492
    case 0:
        return;
#define LCD_REFRESH(depth, func) \
    case depth: \
493 494 495
        col = func(scale_lcd_color(s, (MP_LCD_TEXTCOLOR >> 16) & 0xff), \
                   scale_lcd_color(s, (MP_LCD_TEXTCOLOR >> 8) & 0xff), \
                   scale_lcd_color(s, MP_LCD_TEXTCOLOR & 0xff)); \
J
Jan Kiszka 已提交
496 497 498
        for (x = 0; x < 128; x++) { \
            for (y = 0; y < 64; y++) { \
                if (s->video_ram[x + (y/8)*128] & (1 << (y % 8))) { \
499
                    glue(set_lcd_pixel, depth)(s, x, y, col); \
J
Jan Kiszka 已提交
500
                } else { \
501
                    glue(set_lcd_pixel, depth)(s, x, y, 0); \
J
Jan Kiszka 已提交
502 503 504
                } \
            } \
        } \
505 506 507
        break;
    LCD_REFRESH(8, rgb_to_pixel8)
    LCD_REFRESH(16, rgb_to_pixel16)
508 509
    LCD_REFRESH(32, (is_surface_bgr(s->ds->surface) ?
                     rgb_to_pixel32bgr : rgb_to_pixel32))
510
    default:
P
Paul Brook 已提交
511
        hw_error("unsupported colour depth %i\n",
512
                  ds_get_bits_per_pixel(s->ds));
513
    }
514 515 516 517

    dpy_update(s->ds, 0, 0, 128*3, 64*3);
}

518 519 520 521
static void lcd_invalidate(void *opaque)
{
}

522 523
static void musicpal_lcd_gpio_brigthness_in(void *opaque, int irq, int level)
{
J
Jan Kiszka 已提交
524
    musicpal_lcd_state *s = opaque;
525 526 527 528
    s->brightness &= ~(1 << irq);
    s->brightness |= level << irq;
}

A
Anthony Liguori 已提交
529
static uint32_t musicpal_lcd_read(void *opaque, target_phys_addr_t offset)
530 531 532 533 534 535 536 537 538 539 540 541
{
    musicpal_lcd_state *s = opaque;

    switch (offset) {
    case MP_LCD_IRQCTRL:
        return s->irqctrl;

    default:
        return 0;
    }
}

A
Anthony Liguori 已提交
542
static void musicpal_lcd_write(void *opaque, target_phys_addr_t offset,
543 544 545 546 547 548 549 550 551 552
                               uint32_t value)
{
    musicpal_lcd_state *s = opaque;

    switch (offset) {
    case MP_LCD_IRQCTRL:
        s->irqctrl = value;
        break;

    case MP_LCD_SPICTRL:
J
Jan Kiszka 已提交
553
        if (value == MP_LCD_SPI_DATA || value == MP_LCD_SPI_CMD) {
554
            s->mode = value;
J
Jan Kiszka 已提交
555
        } else {
556
            s->mode = MP_LCD_SPI_INVALID;
J
Jan Kiszka 已提交
557
        }
558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581
        break;

    case MP_LCD_INST:
        if (value >= MP_LCD_INST_SETPAGE0 && value <= MP_LCD_INST_SETPAGE7) {
            s->page = value - MP_LCD_INST_SETPAGE0;
            s->page_off = 0;
        }
        break;

    case MP_LCD_DATA:
        if (s->mode == MP_LCD_SPI_CMD) {
            if (value >= MP_LCD_INST_SETPAGE0 &&
                value <= MP_LCD_INST_SETPAGE7) {
                s->page = value - MP_LCD_INST_SETPAGE0;
                s->page_off = 0;
            }
        } else if (s->mode == MP_LCD_SPI_DATA) {
            s->video_ram[s->page*128 + s->page_off] = value;
            s->page_off = (s->page_off + 1) & 127;
        }
        break;
    }
}

582
static CPUReadMemoryFunc * const musicpal_lcd_readfn[] = {
583 584 585 586 587
    musicpal_lcd_read,
    musicpal_lcd_read,
    musicpal_lcd_read
};

588
static CPUWriteMemoryFunc * const musicpal_lcd_writefn[] = {
589 590 591 592 593
    musicpal_lcd_write,
    musicpal_lcd_write,
    musicpal_lcd_write
};

594
static int musicpal_lcd_init(SysBusDevice *dev)
595
{
P
Paul Brook 已提交
596
    musicpal_lcd_state *s = FROM_SYSBUS(musicpal_lcd_state, dev);
597 598
    int iomemtype;

599 600
    s->brightness = 7;

601
    iomemtype = cpu_register_io_memory(musicpal_lcd_readfn,
602
                                       musicpal_lcd_writefn, s);
P
Paul Brook 已提交
603
    sysbus_init_mmio(dev, MP_LCD_SIZE, iomemtype);
604

605 606 607
    s->ds = graphic_console_init(lcd_refresh, lcd_invalidate,
                                 NULL, NULL, s);
    qemu_console_resize(s->ds, 128*3, 64*3);
608 609

    qdev_init_gpio_in(&dev->qdev, musicpal_lcd_gpio_brigthness_in, 3);
610 611

    return 0;
612 613
}

J
Jan Kiszka 已提交
614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636
static const VMStateDescription musicpal_lcd_vmsd = {
    .name = "musicpal_lcd",
    .version_id = 1,
    .minimum_version_id = 1,
    .minimum_version_id_old = 1,
    .fields = (VMStateField[]) {
        VMSTATE_UINT32(brightness, musicpal_lcd_state),
        VMSTATE_UINT32(mode, musicpal_lcd_state),
        VMSTATE_UINT32(irqctrl, musicpal_lcd_state),
        VMSTATE_UINT32(page, musicpal_lcd_state),
        VMSTATE_UINT32(page_off, musicpal_lcd_state),
        VMSTATE_BUFFER(video_ram, musicpal_lcd_state),
        VMSTATE_END_OF_LIST()
    }
};

static SysBusDeviceInfo musicpal_lcd_info = {
    .init = musicpal_lcd_init,
    .qdev.name = "musicpal_lcd",
    .qdev.size = sizeof(musicpal_lcd_state),
    .qdev.vmsd = &musicpal_lcd_vmsd,
};

637 638 639 640 641 642 643
/* PIC register offsets */
#define MP_PIC_STATUS           0x00
#define MP_PIC_ENABLE_SET       0x08
#define MP_PIC_ENABLE_CLR       0x0C

typedef struct mv88w8618_pic_state
{
P
Paul Brook 已提交
644
    SysBusDevice busdev;
645 646 647 648 649 650 651 652 653 654 655 656 657 658
    uint32_t level;
    uint32_t enabled;
    qemu_irq parent_irq;
} mv88w8618_pic_state;

static void mv88w8618_pic_update(mv88w8618_pic_state *s)
{
    qemu_set_irq(s->parent_irq, (s->level & s->enabled));
}

static void mv88w8618_pic_set_irq(void *opaque, int irq, int level)
{
    mv88w8618_pic_state *s = opaque;

J
Jan Kiszka 已提交
659
    if (level) {
660
        s->level |= 1 << irq;
J
Jan Kiszka 已提交
661
    } else {
662
        s->level &= ~(1 << irq);
J
Jan Kiszka 已提交
663
    }
664 665 666
    mv88w8618_pic_update(s);
}

A
Anthony Liguori 已提交
667
static uint32_t mv88w8618_pic_read(void *opaque, target_phys_addr_t offset)
668 669 670 671 672 673 674 675 676 677 678 679
{
    mv88w8618_pic_state *s = opaque;

    switch (offset) {
    case MP_PIC_STATUS:
        return s->level & s->enabled;

    default:
        return 0;
    }
}

A
Anthony Liguori 已提交
680
static void mv88w8618_pic_write(void *opaque, target_phys_addr_t offset,
681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697
                                uint32_t value)
{
    mv88w8618_pic_state *s = opaque;

    switch (offset) {
    case MP_PIC_ENABLE_SET:
        s->enabled |= value;
        break;

    case MP_PIC_ENABLE_CLR:
        s->enabled &= ~value;
        s->level &= ~value;
        break;
    }
    mv88w8618_pic_update(s);
}

J
Jan Kiszka 已提交
698
static void mv88w8618_pic_reset(DeviceState *d)
699
{
J
Jan Kiszka 已提交
700 701
    mv88w8618_pic_state *s = FROM_SYSBUS(mv88w8618_pic_state,
                                         sysbus_from_qdev(d));
702 703 704 705 706

    s->level = 0;
    s->enabled = 0;
}

707
static CPUReadMemoryFunc * const mv88w8618_pic_readfn[] = {
708 709 710 711 712
    mv88w8618_pic_read,
    mv88w8618_pic_read,
    mv88w8618_pic_read
};

713
static CPUWriteMemoryFunc * const mv88w8618_pic_writefn[] = {
714 715 716 717 718
    mv88w8618_pic_write,
    mv88w8618_pic_write,
    mv88w8618_pic_write
};

719
static int mv88w8618_pic_init(SysBusDevice *dev)
720
{
P
Paul Brook 已提交
721
    mv88w8618_pic_state *s = FROM_SYSBUS(mv88w8618_pic_state, dev);
722 723
    int iomemtype;

P
Paul Brook 已提交
724
    qdev_init_gpio_in(&dev->qdev, mv88w8618_pic_set_irq, 32);
P
Paul Brook 已提交
725
    sysbus_init_irq(dev, &s->parent_irq);
726
    iomemtype = cpu_register_io_memory(mv88w8618_pic_readfn,
727
                                       mv88w8618_pic_writefn, s);
P
Paul Brook 已提交
728
    sysbus_init_mmio(dev, MP_PIC_SIZE, iomemtype);
729
    return 0;
730 731
}

J
Jan Kiszka 已提交
732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751
static const VMStateDescription mv88w8618_pic_vmsd = {
    .name = "mv88w8618_pic",
    .version_id = 1,
    .minimum_version_id = 1,
    .minimum_version_id_old = 1,
    .fields = (VMStateField[]) {
        VMSTATE_UINT32(level, mv88w8618_pic_state),
        VMSTATE_UINT32(enabled, mv88w8618_pic_state),
        VMSTATE_END_OF_LIST()
    }
};

static SysBusDeviceInfo mv88w8618_pic_info = {
    .init = mv88w8618_pic_init,
    .qdev.name = "mv88w8618_pic",
    .qdev.size = sizeof(mv88w8618_pic_state),
    .qdev.reset = mv88w8618_pic_reset,
    .qdev.vmsd = &mv88w8618_pic_vmsd,
};

752 753 754 755 756 757 758 759 760 761 762 763 764 765
/* PIT register offsets */
#define MP_PIT_TIMER1_LENGTH    0x00
/* ... */
#define MP_PIT_TIMER4_LENGTH    0x0C
#define MP_PIT_CONTROL          0x10
#define MP_PIT_TIMER1_VALUE     0x14
/* ... */
#define MP_PIT_TIMER4_VALUE     0x20
#define MP_BOARD_RESET          0x34

/* Magic board reset value (probably some watchdog behind it) */
#define MP_BOARD_RESET_MAGIC    0x10000

typedef struct mv88w8618_timer_state {
P
Paul Brook 已提交
766
    ptimer_state *ptimer;
767 768 769 770 771 772
    uint32_t limit;
    int freq;
    qemu_irq irq;
} mv88w8618_timer_state;

typedef struct mv88w8618_pit_state {
P
Paul Brook 已提交
773 774
    SysBusDevice busdev;
    mv88w8618_timer_state timer[4];
775 776 777 778 779 780 781 782 783
} mv88w8618_pit_state;

static void mv88w8618_timer_tick(void *opaque)
{
    mv88w8618_timer_state *s = opaque;

    qemu_irq_raise(s->irq);
}

P
Paul Brook 已提交
784 785
static void mv88w8618_timer_init(SysBusDevice *dev, mv88w8618_timer_state *s,
                                 uint32_t freq)
786 787 788
{
    QEMUBH *bh;

P
Paul Brook 已提交
789
    sysbus_init_irq(dev, &s->irq);
790 791 792
    s->freq = freq;

    bh = qemu_bh_new(mv88w8618_timer_tick, s);
P
Paul Brook 已提交
793
    s->ptimer = ptimer_init(bh);
794 795
}

A
Anthony Liguori 已提交
796
static uint32_t mv88w8618_pit_read(void *opaque, target_phys_addr_t offset)
797 798 799 800 801 802
{
    mv88w8618_pit_state *s = opaque;
    mv88w8618_timer_state *t;

    switch (offset) {
    case MP_PIT_TIMER1_VALUE ... MP_PIT_TIMER4_VALUE:
P
Paul Brook 已提交
803 804
        t = &s->timer[(offset-MP_PIT_TIMER1_VALUE) >> 2];
        return ptimer_get_count(t->ptimer);
805 806 807 808 809 810

    default:
        return 0;
    }
}

A
Anthony Liguori 已提交
811
static void mv88w8618_pit_write(void *opaque, target_phys_addr_t offset,
812 813 814 815 816 817 818 819
                                uint32_t value)
{
    mv88w8618_pit_state *s = opaque;
    mv88w8618_timer_state *t;
    int i;

    switch (offset) {
    case MP_PIT_TIMER1_LENGTH ... MP_PIT_TIMER4_LENGTH:
P
Paul Brook 已提交
820
        t = &s->timer[offset >> 2];
821
        t->limit = value;
822 823 824 825 826
        if (t->limit > 0) {
            ptimer_set_limit(t->ptimer, t->limit, 1);
        } else {
            ptimer_stop(t->ptimer);
        }
827 828 829 830
        break;

    case MP_PIT_CONTROL:
        for (i = 0; i < 4; i++) {
831 832
            t = &s->timer[i];
            if (value & 0xf && t->limit > 0) {
P
Paul Brook 已提交
833 834 835
                ptimer_set_limit(t->ptimer, t->limit, 0);
                ptimer_set_freq(t->ptimer, t->freq);
                ptimer_run(t->ptimer, 0);
836 837
            } else {
                ptimer_stop(t->ptimer);
838 839 840 841 842 843
            }
            value >>= 4;
        }
        break;

    case MP_BOARD_RESET:
J
Jan Kiszka 已提交
844
        if (value == MP_BOARD_RESET_MAGIC) {
845
            qemu_system_reset_request();
J
Jan Kiszka 已提交
846
        }
847 848 849 850
        break;
    }
}

J
Jan Kiszka 已提交
851
static void mv88w8618_pit_reset(DeviceState *d)
852
{
J
Jan Kiszka 已提交
853 854
    mv88w8618_pit_state *s = FROM_SYSBUS(mv88w8618_pit_state,
                                         sysbus_from_qdev(d));
855 856 857 858 859 860 861 862
    int i;

    for (i = 0; i < 4; i++) {
        ptimer_stop(s->timer[i].ptimer);
        s->timer[i].limit = 0;
    }
}

863
static CPUReadMemoryFunc * const mv88w8618_pit_readfn[] = {
864 865 866 867 868
    mv88w8618_pit_read,
    mv88w8618_pit_read,
    mv88w8618_pit_read
};

869
static CPUWriteMemoryFunc * const mv88w8618_pit_writefn[] = {
870 871 872 873 874
    mv88w8618_pit_write,
    mv88w8618_pit_write,
    mv88w8618_pit_write
};

875
static int mv88w8618_pit_init(SysBusDevice *dev)
876 877
{
    int iomemtype;
P
Paul Brook 已提交
878 879
    mv88w8618_pit_state *s = FROM_SYSBUS(mv88w8618_pit_state, dev);
    int i;
880 881 882

    /* Letting them all run at 1 MHz is likely just a pragmatic
     * simplification. */
P
Paul Brook 已提交
883 884 885
    for (i = 0; i < 4; i++) {
        mv88w8618_timer_init(dev, &s->timer[i], 1000000);
    }
886

887
    iomemtype = cpu_register_io_memory(mv88w8618_pit_readfn,
888
                                       mv88w8618_pit_writefn, s);
P
Paul Brook 已提交
889
    sysbus_init_mmio(dev, MP_PIT_SIZE, iomemtype);
890
    return 0;
891 892
}

J
Jan Kiszka 已提交
893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916
static const VMStateDescription mv88w8618_timer_vmsd = {
    .name = "timer",
    .version_id = 1,
    .minimum_version_id = 1,
    .minimum_version_id_old = 1,
    .fields = (VMStateField[]) {
        VMSTATE_PTIMER(ptimer, mv88w8618_timer_state),
        VMSTATE_UINT32(limit, mv88w8618_timer_state),
        VMSTATE_END_OF_LIST()
    }
};

static const VMStateDescription mv88w8618_pit_vmsd = {
    .name = "mv88w8618_pit",
    .version_id = 1,
    .minimum_version_id = 1,
    .minimum_version_id_old = 1,
    .fields = (VMStateField[]) {
        VMSTATE_STRUCT_ARRAY(timer, mv88w8618_pit_state, 4, 1,
                             mv88w8618_timer_vmsd, mv88w8618_timer_state),
        VMSTATE_END_OF_LIST()
    }
};

917 918 919 920 921
static SysBusDeviceInfo mv88w8618_pit_info = {
    .init = mv88w8618_pit_init,
    .qdev.name  = "mv88w8618_pit",
    .qdev.size  = sizeof(mv88w8618_pit_state),
    .qdev.reset = mv88w8618_pit_reset,
J
Jan Kiszka 已提交
922
    .qdev.vmsd  = &mv88w8618_pit_vmsd,
923 924
};

925 926 927 928
/* Flash config register offsets */
#define MP_FLASHCFG_CFGR0    0x04

typedef struct mv88w8618_flashcfg_state {
P
Paul Brook 已提交
929
    SysBusDevice busdev;
930 931 932 933
    uint32_t cfgr0;
} mv88w8618_flashcfg_state;

static uint32_t mv88w8618_flashcfg_read(void *opaque,
A
Anthony Liguori 已提交
934
                                        target_phys_addr_t offset)
935 936 937 938 939 940 941 942 943 944 945 946
{
    mv88w8618_flashcfg_state *s = opaque;

    switch (offset) {
    case MP_FLASHCFG_CFGR0:
        return s->cfgr0;

    default:
        return 0;
    }
}

A
Anthony Liguori 已提交
947
static void mv88w8618_flashcfg_write(void *opaque, target_phys_addr_t offset,
948 949 950 951 952 953 954 955 956 957 958
                                     uint32_t value)
{
    mv88w8618_flashcfg_state *s = opaque;

    switch (offset) {
    case MP_FLASHCFG_CFGR0:
        s->cfgr0 = value;
        break;
    }
}

959
static CPUReadMemoryFunc * const mv88w8618_flashcfg_readfn[] = {
960 961 962 963 964
    mv88w8618_flashcfg_read,
    mv88w8618_flashcfg_read,
    mv88w8618_flashcfg_read
};

965
static CPUWriteMemoryFunc * const mv88w8618_flashcfg_writefn[] = {
966 967 968 969 970
    mv88w8618_flashcfg_write,
    mv88w8618_flashcfg_write,
    mv88w8618_flashcfg_write
};

971
static int mv88w8618_flashcfg_init(SysBusDevice *dev)
972 973
{
    int iomemtype;
P
Paul Brook 已提交
974
    mv88w8618_flashcfg_state *s = FROM_SYSBUS(mv88w8618_flashcfg_state, dev);
975 976

    s->cfgr0 = 0xfffe4285; /* Default as set by U-Boot for 8 MB flash */
977
    iomemtype = cpu_register_io_memory(mv88w8618_flashcfg_readfn,
J
Jan Kiszka 已提交
978
                                       mv88w8618_flashcfg_writefn, s);
P
Paul Brook 已提交
979
    sysbus_init_mmio(dev, MP_FLASHCFG_SIZE, iomemtype);
980
    return 0;
981 982
}

J
Jan Kiszka 已提交
983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000
static const VMStateDescription mv88w8618_flashcfg_vmsd = {
    .name = "mv88w8618_flashcfg",
    .version_id = 1,
    .minimum_version_id = 1,
    .minimum_version_id_old = 1,
    .fields = (VMStateField[]) {
        VMSTATE_UINT32(cfgr0, mv88w8618_flashcfg_state),
        VMSTATE_END_OF_LIST()
    }
};

static SysBusDeviceInfo mv88w8618_flashcfg_info = {
    .init = mv88w8618_flashcfg_init,
    .qdev.name  = "mv88w8618_flashcfg",
    .qdev.size  = sizeof(mv88w8618_flashcfg_state),
    .qdev.vmsd  = &mv88w8618_flashcfg_vmsd,
};

1001 1002 1003 1004 1005
/* Misc register offsets */
#define MP_MISC_BOARD_REVISION  0x18

#define MP_BOARD_REVISION       0x31

A
Anthony Liguori 已提交
1006
static uint32_t musicpal_misc_read(void *opaque, target_phys_addr_t offset)
1007 1008 1009 1010 1011 1012 1013 1014 1015 1016
{
    switch (offset) {
    case MP_MISC_BOARD_REVISION:
        return MP_BOARD_REVISION;

    default:
        return 0;
    }
}

A
Anthony Liguori 已提交
1017
static void musicpal_misc_write(void *opaque, target_phys_addr_t offset,
1018 1019 1020 1021
                                uint32_t value)
{
}

1022
static CPUReadMemoryFunc * const musicpal_misc_readfn[] = {
1023 1024 1025 1026 1027
    musicpal_misc_read,
    musicpal_misc_read,
    musicpal_misc_read,
};

1028
static CPUWriteMemoryFunc * const musicpal_misc_writefn[] = {
1029 1030 1031 1032 1033 1034 1035 1036 1037
    musicpal_misc_write,
    musicpal_misc_write,
    musicpal_misc_write,
};

static void musicpal_misc_init(void)
{
    int iomemtype;

1038
    iomemtype = cpu_register_io_memory(musicpal_misc_readfn,
1039 1040 1041 1042 1043 1044 1045 1046
                                       musicpal_misc_writefn, NULL);
    cpu_register_physical_memory(MP_MISC_BASE, MP_MISC_SIZE, iomemtype);
}

/* WLAN register offsets */
#define MP_WLAN_MAGIC1          0x11c
#define MP_WLAN_MAGIC2          0x124

A
Anthony Liguori 已提交
1047
static uint32_t mv88w8618_wlan_read(void *opaque, target_phys_addr_t offset)
1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061
{
    switch (offset) {
    /* Workaround to allow loading the binary-only wlandrv.ko crap
     * from the original Freecom firmware. */
    case MP_WLAN_MAGIC1:
        return ~3;
    case MP_WLAN_MAGIC2:
        return -1;

    default:
        return 0;
    }
}

A
Anthony Liguori 已提交
1062
static void mv88w8618_wlan_write(void *opaque, target_phys_addr_t offset,
1063 1064 1065 1066
                                 uint32_t value)
{
}

1067
static CPUReadMemoryFunc * const mv88w8618_wlan_readfn[] = {
1068 1069 1070 1071 1072
    mv88w8618_wlan_read,
    mv88w8618_wlan_read,
    mv88w8618_wlan_read,
};

1073
static CPUWriteMemoryFunc * const mv88w8618_wlan_writefn[] = {
1074 1075 1076 1077 1078
    mv88w8618_wlan_write,
    mv88w8618_wlan_write,
    mv88w8618_wlan_write,
};

1079
static int mv88w8618_wlan_init(SysBusDevice *dev)
1080 1081
{
    int iomemtype;
1082

1083
    iomemtype = cpu_register_io_memory(mv88w8618_wlan_readfn,
1084
                                       mv88w8618_wlan_writefn, NULL);
P
Paul Brook 已提交
1085
    sysbus_init_mmio(dev, MP_WLAN_SIZE, iomemtype);
1086
    return 0;
1087
}
1088

1089 1090 1091 1092
/* GPIO register offsets */
#define MP_GPIO_OE_LO           0x008
#define MP_GPIO_OUT_LO          0x00c
#define MP_GPIO_IN_LO           0x010
J
Jan Kiszka 已提交
1093 1094
#define MP_GPIO_IER_LO          0x014
#define MP_GPIO_IMR_LO          0x018
1095 1096 1097 1098
#define MP_GPIO_ISR_LO          0x020
#define MP_GPIO_OE_HI           0x508
#define MP_GPIO_OUT_HI          0x50c
#define MP_GPIO_IN_HI           0x510
J
Jan Kiszka 已提交
1099 1100
#define MP_GPIO_IER_HI          0x514
#define MP_GPIO_IMR_HI          0x518
1101
#define MP_GPIO_ISR_HI          0x520
1102 1103 1104 1105 1106 1107 1108 1109 1110

/* GPIO bits & masks */
#define MP_GPIO_LCD_BRIGHTNESS  0x00070000
#define MP_GPIO_I2C_DATA_BIT    29
#define MP_GPIO_I2C_CLOCK_BIT   30

/* LCD brightness bits in GPIO_OE_HI */
#define MP_OE_LCD_BRIGHTNESS    0x0007

1111 1112 1113 1114 1115
typedef struct musicpal_gpio_state {
    SysBusDevice busdev;
    uint32_t lcd_brightness;
    uint32_t out_state;
    uint32_t in_state;
J
Jan Kiszka 已提交
1116 1117
    uint32_t ier;
    uint32_t imr;
1118 1119
    uint32_t isr;
    qemu_irq irq;
J
Jan Kiszka 已提交
1120
    qemu_irq out[5]; /* 3 brightness out + 2 lcd (data and clock ) */
1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162
} musicpal_gpio_state;

static void musicpal_gpio_brightness_update(musicpal_gpio_state *s) {
    int i;
    uint32_t brightness;

    /* compute brightness ratio */
    switch (s->lcd_brightness) {
    case 0x00000007:
        brightness = 0;
        break;

    case 0x00020000:
        brightness = 1;
        break;

    case 0x00020001:
        brightness = 2;
        break;

    case 0x00040000:
        brightness = 3;
        break;

    case 0x00010006:
        brightness = 4;
        break;

    case 0x00020005:
        brightness = 5;
        break;

    case 0x00040003:
        brightness = 6;
        break;

    case 0x00030004:
    default:
        brightness = 7;
    }

    /* set lcd brightness GPIOs  */
J
Jan Kiszka 已提交
1163
    for (i = 0; i <= 2; i++) {
1164
        qemu_set_irq(s->out[i], (brightness >> i) & 1);
J
Jan Kiszka 已提交
1165
    }
1166 1167
}

J
Jan Kiszka 已提交
1168
static void musicpal_gpio_pin_event(void *opaque, int pin, int level)
1169
{
J
Jan Kiszka 已提交
1170
    musicpal_gpio_state *s = opaque;
J
Jan Kiszka 已提交
1171 1172 1173
    uint32_t mask = 1 << pin;
    uint32_t delta = level << pin;
    uint32_t old = s->in_state & mask;
1174

J
Jan Kiszka 已提交
1175 1176
    s->in_state &= ~mask;
    s->in_state |= delta;
1177

J
Jan Kiszka 已提交
1178 1179 1180 1181
    if ((old ^ delta) &&
        ((level && (s->imr & mask)) || (!level && (s->ier & mask)))) {
        s->isr = mask;
        qemu_irq_raise(s->irq);
1182 1183 1184
    }
}

A
Anthony Liguori 已提交
1185
static uint32_t musicpal_gpio_read(void *opaque, target_phys_addr_t offset)
1186
{
J
Jan Kiszka 已提交
1187
    musicpal_gpio_state *s = opaque;
1188

1189 1190
    switch (offset) {
    case MP_GPIO_OE_HI: /* used for LCD brightness control */
1191
        return s->lcd_brightness & MP_OE_LCD_BRIGHTNESS;
1192 1193

    case MP_GPIO_OUT_LO:
1194
        return s->out_state & 0xFFFF;
1195
    case MP_GPIO_OUT_HI:
1196
        return s->out_state >> 16;
1197 1198

    case MP_GPIO_IN_LO:
1199
        return s->in_state & 0xFFFF;
1200
    case MP_GPIO_IN_HI:
1201
        return s->in_state >> 16;
1202

J
Jan Kiszka 已提交
1203 1204 1205 1206 1207 1208 1209 1210 1211 1212
    case MP_GPIO_IER_LO:
        return s->ier & 0xFFFF;
    case MP_GPIO_IER_HI:
        return s->ier >> 16;

    case MP_GPIO_IMR_LO:
        return s->imr & 0xFFFF;
    case MP_GPIO_IMR_HI:
        return s->imr >> 16;

1213
    case MP_GPIO_ISR_LO:
1214
        return s->isr & 0xFFFF;
1215
    case MP_GPIO_ISR_HI:
1216
        return s->isr >> 16;
1217 1218 1219 1220 1221 1222

    default:
        return 0;
    }
}

A
Anthony Liguori 已提交
1223
static void musicpal_gpio_write(void *opaque, target_phys_addr_t offset,
1224
                                uint32_t value)
1225
{
J
Jan Kiszka 已提交
1226
    musicpal_gpio_state *s = opaque;
1227 1228
    switch (offset) {
    case MP_GPIO_OE_HI: /* used for LCD brightness control */
1229
        s->lcd_brightness = (s->lcd_brightness & MP_GPIO_LCD_BRIGHTNESS) |
1230
                         (value & MP_OE_LCD_BRIGHTNESS);
1231
        musicpal_gpio_brightness_update(s);
1232 1233 1234
        break;

    case MP_GPIO_OUT_LO:
1235
        s->out_state = (s->out_state & 0xFFFF0000) | (value & 0xFFFF);
1236 1237
        break;
    case MP_GPIO_OUT_HI:
1238 1239 1240 1241
        s->out_state = (s->out_state & 0xFFFF) | (value << 16);
        s->lcd_brightness = (s->lcd_brightness & 0xFFFF) |
                            (s->out_state & MP_GPIO_LCD_BRIGHTNESS);
        musicpal_gpio_brightness_update(s);
1242 1243
        qemu_set_irq(s->out[3], (s->out_state >> MP_GPIO_I2C_DATA_BIT) & 1);
        qemu_set_irq(s->out[4], (s->out_state >> MP_GPIO_I2C_CLOCK_BIT) & 1);
1244 1245
        break;

J
Jan Kiszka 已提交
1246 1247 1248 1249 1250 1251 1252 1253 1254 1255 1256 1257 1258
    case MP_GPIO_IER_LO:
        s->ier = (s->ier & 0xFFFF0000) | (value & 0xFFFF);
        break;
    case MP_GPIO_IER_HI:
        s->ier = (s->ier & 0xFFFF) | (value << 16);
        break;

    case MP_GPIO_IMR_LO:
        s->imr = (s->imr & 0xFFFF0000) | (value & 0xFFFF);
        break;
    case MP_GPIO_IMR_HI:
        s->imr = (s->imr & 0xFFFF) | (value << 16);
        break;
1259 1260 1261
    }
}

1262
static CPUReadMemoryFunc * const musicpal_gpio_readfn[] = {
1263 1264 1265 1266 1267
    musicpal_gpio_read,
    musicpal_gpio_read,
    musicpal_gpio_read,
};

1268
static CPUWriteMemoryFunc * const musicpal_gpio_writefn[] = {
1269 1270 1271 1272 1273
    musicpal_gpio_write,
    musicpal_gpio_write,
    musicpal_gpio_write,
};

J
Jan Kiszka 已提交
1274
static void musicpal_gpio_reset(DeviceState *d)
1275
{
J
Jan Kiszka 已提交
1276 1277
    musicpal_gpio_state *s = FROM_SYSBUS(musicpal_gpio_state,
                                         sysbus_from_qdev(d));
J
Jan Kiszka 已提交
1278 1279 1280

    s->lcd_brightness = 0;
    s->out_state = 0;
1281
    s->in_state = 0xffffffff;
J
Jan Kiszka 已提交
1282 1283
    s->ier = 0;
    s->imr = 0;
1284 1285 1286
    s->isr = 0;
}

1287
static int musicpal_gpio_init(SysBusDevice *dev)
1288 1289
{
    musicpal_gpio_state *s = FROM_SYSBUS(musicpal_gpio_state, dev);
1290 1291
    int iomemtype;

1292 1293
    sysbus_init_irq(dev, &s->irq);

1294
    iomemtype = cpu_register_io_memory(musicpal_gpio_readfn,
1295 1296 1297
                                       musicpal_gpio_writefn, s);
    sysbus_init_mmio(dev, MP_GPIO_SIZE, iomemtype);

J
Jan Kiszka 已提交
1298 1299 1300
    qdev_init_gpio_out(&dev->qdev, s->out, ARRAY_SIZE(s->out));

    qdev_init_gpio_in(&dev->qdev, musicpal_gpio_pin_event, 32);
1301 1302

    return 0;
1303 1304
}

J
Jan Kiszka 已提交
1305 1306 1307 1308 1309 1310 1311 1312 1313 1314 1315 1316 1317 1318 1319 1320
static const VMStateDescription musicpal_gpio_vmsd = {
    .name = "musicpal_gpio",
    .version_id = 1,
    .minimum_version_id = 1,
    .minimum_version_id_old = 1,
    .fields = (VMStateField[]) {
        VMSTATE_UINT32(lcd_brightness, musicpal_gpio_state),
        VMSTATE_UINT32(out_state, musicpal_gpio_state),
        VMSTATE_UINT32(in_state, musicpal_gpio_state),
        VMSTATE_UINT32(ier, musicpal_gpio_state),
        VMSTATE_UINT32(imr, musicpal_gpio_state),
        VMSTATE_UINT32(isr, musicpal_gpio_state),
        VMSTATE_END_OF_LIST()
    }
};

J
Jan Kiszka 已提交
1321 1322 1323 1324 1325
static SysBusDeviceInfo musicpal_gpio_info = {
    .init = musicpal_gpio_init,
    .qdev.name  = "musicpal_gpio",
    .qdev.size  = sizeof(musicpal_gpio_state),
    .qdev.reset = musicpal_gpio_reset,
J
Jan Kiszka 已提交
1326
    .qdev.vmsd  = &musicpal_gpio_vmsd,
J
Jan Kiszka 已提交
1327 1328
};

1329
/* Keyboard codes & masks */
1330
#define KEY_RELEASED            0x80
1331 1332 1333 1334 1335 1336 1337 1338 1339 1340 1341 1342 1343
#define KEY_CODE                0x7f

#define KEYCODE_TAB             0x0f
#define KEYCODE_ENTER           0x1c
#define KEYCODE_F               0x21
#define KEYCODE_M               0x32

#define KEYCODE_EXTENDED        0xe0
#define KEYCODE_UP              0x48
#define KEYCODE_DOWN            0x50
#define KEYCODE_LEFT            0x4b
#define KEYCODE_RIGHT           0x4d

J
Jan Kiszka 已提交
1344
#define MP_KEY_WHEEL_VOL       (1 << 0)
1345 1346 1347 1348 1349 1350 1351 1352 1353 1354 1355
#define MP_KEY_WHEEL_VOL_INV   (1 << 1)
#define MP_KEY_WHEEL_NAV       (1 << 2)
#define MP_KEY_WHEEL_NAV_INV   (1 << 3)
#define MP_KEY_BTN_FAVORITS    (1 << 4)
#define MP_KEY_BTN_MENU        (1 << 5)
#define MP_KEY_BTN_VOLUME      (1 << 6)
#define MP_KEY_BTN_NAVIGATION  (1 << 7)

typedef struct musicpal_key_state {
    SysBusDevice busdev;
    uint32_t kbd_extended;
J
Jan Kiszka 已提交
1356 1357
    uint32_t pressed_keys;
    qemu_irq out[8];
1358 1359
} musicpal_key_state;

1360 1361
static void musicpal_key_event(void *opaque, int keycode)
{
J
Jan Kiszka 已提交
1362
    musicpal_key_state *s = opaque;
1363
    uint32_t event = 0;
1364
    int i;
1365 1366

    if (keycode == KEYCODE_EXTENDED) {
1367
        s->kbd_extended = 1;
1368 1369 1370
        return;
    }

J
Jan Kiszka 已提交
1371
    if (s->kbd_extended) {
1372 1373
        switch (keycode & KEY_CODE) {
        case KEYCODE_UP:
1374
            event = MP_KEY_WHEEL_NAV | MP_KEY_WHEEL_NAV_INV;
1375 1376 1377
            break;

        case KEYCODE_DOWN:
1378
            event = MP_KEY_WHEEL_NAV;
1379 1380 1381
            break;

        case KEYCODE_LEFT:
1382
            event = MP_KEY_WHEEL_VOL | MP_KEY_WHEEL_VOL_INV;
1383 1384 1385
            break;

        case KEYCODE_RIGHT:
1386
            event = MP_KEY_WHEEL_VOL;
1387 1388
            break;
        }
J
Jan Kiszka 已提交
1389
    } else {
1390 1391
        switch (keycode & KEY_CODE) {
        case KEYCODE_F:
1392
            event = MP_KEY_BTN_FAVORITS;
1393 1394 1395
            break;

        case KEYCODE_TAB:
1396
            event = MP_KEY_BTN_VOLUME;
1397 1398 1399
            break;

        case KEYCODE_ENTER:
1400
            event = MP_KEY_BTN_NAVIGATION;
1401 1402 1403
            break;

        case KEYCODE_M:
1404
            event = MP_KEY_BTN_MENU;
1405 1406
            break;
        }
1407
        /* Do not repeat already pressed buttons */
J
Jan Kiszka 已提交
1408
        if (!(keycode & KEY_RELEASED) && (s->pressed_keys & event)) {
1409
            event = 0;
J
Jan Kiszka 已提交
1410
        }
1411
    }
1412

1413
    if (event) {
J
Jan Kiszka 已提交
1414 1415 1416 1417 1418 1419 1420 1421 1422 1423 1424 1425 1426
        /* Raise GPIO pin first if repeating a key */
        if (!(keycode & KEY_RELEASED) && (s->pressed_keys & event)) {
            for (i = 0; i <= 7; i++) {
                if (event & (1 << i)) {
                    qemu_set_irq(s->out[i], 1);
                }
            }
        }
        for (i = 0; i <= 7; i++) {
            if (event & (1 << i)) {
                qemu_set_irq(s->out[i], !!(keycode & KEY_RELEASED));
            }
        }
1427
        if (keycode & KEY_RELEASED) {
J
Jan Kiszka 已提交
1428
            s->pressed_keys &= ~event;
1429
        } else {
J
Jan Kiszka 已提交
1430
            s->pressed_keys |= event;
1431
        }
1432 1433
    }

1434 1435 1436
    s->kbd_extended = 0;
}

1437
static int musicpal_key_init(SysBusDevice *dev)
1438 1439 1440 1441 1442 1443
{
    musicpal_key_state *s = FROM_SYSBUS(musicpal_key_state, dev);

    sysbus_init_mmio(dev, 0x0, 0);

    s->kbd_extended = 0;
J
Jan Kiszka 已提交
1444
    s->pressed_keys = 0;
1445

J
Jan Kiszka 已提交
1446
    qdev_init_gpio_out(&dev->qdev, s->out, ARRAY_SIZE(s->out));
1447 1448

    qemu_add_kbd_event_handler(musicpal_key_event, s);
1449 1450

    return 0;
1451 1452
}

J
Jan Kiszka 已提交
1453 1454 1455 1456 1457 1458 1459 1460 1461 1462 1463 1464 1465 1466 1467 1468 1469 1470 1471
static const VMStateDescription musicpal_key_vmsd = {
    .name = "musicpal_key",
    .version_id = 1,
    .minimum_version_id = 1,
    .minimum_version_id_old = 1,
    .fields = (VMStateField[]) {
        VMSTATE_UINT32(kbd_extended, musicpal_key_state),
        VMSTATE_UINT32(pressed_keys, musicpal_key_state),
        VMSTATE_END_OF_LIST()
    }
};

static SysBusDeviceInfo musicpal_key_info = {
    .init = musicpal_key_init,
    .qdev.name  = "musicpal_key",
    .qdev.size  = sizeof(musicpal_key_state),
    .qdev.vmsd  = &musicpal_key_vmsd,
};

1472 1473 1474 1475 1476
static struct arm_boot_info musicpal_binfo = {
    .loader_start = 0x0,
    .board_id = 0x20e,
};

A
Anthony Liguori 已提交
1477
static void musicpal_init(ram_addr_t ram_size,
1478
               const char *boot_device,
1479 1480 1481 1482
               const char *kernel_filename, const char *kernel_cmdline,
               const char *initrd_filename, const char *cpu_model)
{
    CPUState *env;
P
Paul Brook 已提交
1483 1484 1485
    qemu_irq *cpu_pic;
    qemu_irq pic[32];
    DeviceState *dev;
1486
    DeviceState *i2c_dev;
1487 1488
    DeviceState *lcd_dev;
    DeviceState *key_dev;
1489 1490 1491
    DeviceState *wm8750_dev;
    SysBusDevice *s;
    i2c_bus *i2c;
P
Paul Brook 已提交
1492
    int i;
1493
    unsigned long flash_size;
G
Gerd Hoffmann 已提交
1494
    DriveInfo *dinfo;
A
Anthony Liguori 已提交
1495
    ram_addr_t sram_off;
1496

J
Jan Kiszka 已提交
1497
    if (!cpu_model) {
1498
        cpu_model = "arm926";
J
Jan Kiszka 已提交
1499
    }
1500 1501 1502 1503 1504
    env = cpu_init(cpu_model);
    if (!env) {
        fprintf(stderr, "Unable to find CPU definition\n");
        exit(1);
    }
P
Paul Brook 已提交
1505
    cpu_pic = arm_pic_init_cpu(env);
1506 1507 1508 1509 1510 1511 1512 1513

    /* For now we use a fixed - the original - RAM size */
    cpu_register_physical_memory(0, MP_RAM_DEFAULT_SIZE,
                                 qemu_ram_alloc(MP_RAM_DEFAULT_SIZE));

    sram_off = qemu_ram_alloc(MP_SRAM_SIZE);
    cpu_register_physical_memory(MP_SRAM_BASE, MP_SRAM_SIZE, sram_off);

P
Paul Brook 已提交
1514 1515 1516
    dev = sysbus_create_simple("mv88w8618_pic", MP_PIC_BASE,
                               cpu_pic[ARM_PIC_CPU_IRQ]);
    for (i = 0; i < 32; i++) {
P
Paul Brook 已提交
1517
        pic[i] = qdev_get_gpio_in(dev, i);
P
Paul Brook 已提交
1518 1519 1520 1521
    }
    sysbus_create_varargs("mv88w8618_pit", MP_PIT_BASE, pic[MP_TIMER1_IRQ],
                          pic[MP_TIMER2_IRQ], pic[MP_TIMER3_IRQ],
                          pic[MP_TIMER4_IRQ], NULL);
1522

J
Jan Kiszka 已提交
1523
    if (serial_hds[0]) {
B
Blue Swirl 已提交
1524
#ifdef TARGET_WORDS_BIGENDIAN
A
aurel32 已提交
1525
        serial_mm_init(MP_UART1_BASE, 2, pic[MP_UART1_IRQ], 1825000,
B
Blue Swirl 已提交
1526 1527 1528 1529 1530
                       serial_hds[0], 1, 1);
#else
        serial_mm_init(MP_UART1_BASE, 2, pic[MP_UART1_IRQ], 1825000,
                       serial_hds[0], 1, 0);
#endif
J
Jan Kiszka 已提交
1531 1532
    }
    if (serial_hds[1]) {
B
Blue Swirl 已提交
1533
#ifdef TARGET_WORDS_BIGENDIAN
A
aurel32 已提交
1534
        serial_mm_init(MP_UART2_BASE, 2, pic[MP_UART2_IRQ], 1825000,
B
Blue Swirl 已提交
1535 1536 1537 1538 1539
                       serial_hds[1], 1, 1);
#else
        serial_mm_init(MP_UART2_BASE, 2, pic[MP_UART2_IRQ], 1825000,
                       serial_hds[1], 1, 0);
#endif
J
Jan Kiszka 已提交
1540
    }
1541 1542

    /* Register flash */
G
Gerd Hoffmann 已提交
1543 1544 1545
    dinfo = drive_get(IF_PFLASH, 0, 0);
    if (dinfo) {
        flash_size = bdrv_getlength(dinfo->bdrv);
1546 1547 1548 1549 1550 1551 1552 1553 1554 1555 1556
        if (flash_size != 8*1024*1024 && flash_size != 16*1024*1024 &&
            flash_size != 32*1024*1024) {
            fprintf(stderr, "Invalid flash image size\n");
            exit(1);
        }

        /*
         * The original U-Boot accesses the flash at 0xFE000000 instead of
         * 0xFF800000 (if there is 8 MB flash). So remap flash access if the
         * image is smaller than 32 MB.
         */
B
Blue Swirl 已提交
1557
#ifdef TARGET_WORDS_BIGENDIAN
1558
        pflash_cfi02_register(0-MP_FLASH_SIZE_MAX, qemu_ram_alloc(flash_size),
G
Gerd Hoffmann 已提交
1559
                              dinfo->bdrv, 0x10000,
1560 1561 1562
                              (flash_size + 0xffff) >> 16,
                              MP_FLASH_SIZE_MAX / flash_size,
                              2, 0x00BF, 0x236D, 0x0000, 0x0000,
B
Blue Swirl 已提交
1563 1564 1565 1566 1567 1568 1569 1570 1571 1572
                              0x5555, 0x2AAA, 1);
#else
        pflash_cfi02_register(0-MP_FLASH_SIZE_MAX, qemu_ram_alloc(flash_size),
                              dinfo->bdrv, 0x10000,
                              (flash_size + 0xffff) >> 16,
                              MP_FLASH_SIZE_MAX / flash_size,
                              2, 0x00BF, 0x236D, 0x0000, 0x0000,
                              0x5555, 0x2AAA, 0);
#endif

1573
    }
P
Paul Brook 已提交
1574
    sysbus_create_simple("mv88w8618_flashcfg", MP_FLASHCFG_BASE, NULL);
1575

P
Paul Brook 已提交
1576 1577
    qemu_check_nic_model(&nd_table[0], "mv88w8618");
    dev = qdev_create(NULL, "mv88w8618_eth");
1578
    qdev_set_nic_properties(dev, &nd_table[0]);
M
Markus Armbruster 已提交
1579
    qdev_init_nofail(dev);
P
Paul Brook 已提交
1580 1581
    sysbus_mmio_map(sysbus_from_qdev(dev), 0, MP_ETH_BASE);
    sysbus_connect_irq(sysbus_from_qdev(dev), 0, pic[MP_ETH_IRQ]);
1582

P
Paul Brook 已提交
1583
    sysbus_create_simple("mv88w8618_wlan", MP_WLAN_BASE, NULL);
1584 1585

    musicpal_misc_init();
1586 1587

    dev = sysbus_create_simple("musicpal_gpio", MP_GPIO_BASE, pic[MP_GPIO_IRQ]);
P
Paul Brook 已提交
1588
    i2c_dev = sysbus_create_simple("gpio_i2c", 0, NULL);
1589 1590
    i2c = (i2c_bus *)qdev_get_child_bus(i2c_dev, "i2c");

1591 1592 1593
    lcd_dev = sysbus_create_simple("musicpal_lcd", MP_LCD_BASE, NULL);
    key_dev = sysbus_create_simple("musicpal_key", 0, NULL);

1594
    /* I2C read data */
J
Jan Kiszka 已提交
1595 1596
    qdev_connect_gpio_out(i2c_dev, 0,
                          qdev_get_gpio_in(dev, MP_GPIO_I2C_DATA_BIT));
1597 1598 1599 1600 1601
    /* I2C data */
    qdev_connect_gpio_out(dev, 3, qdev_get_gpio_in(i2c_dev, 0));
    /* I2C clock */
    qdev_connect_gpio_out(dev, 4, qdev_get_gpio_in(i2c_dev, 1));

J
Jan Kiszka 已提交
1602
    for (i = 0; i < 3; i++) {
1603
        qdev_connect_gpio_out(dev, i, qdev_get_gpio_in(lcd_dev, i));
J
Jan Kiszka 已提交
1604
    }
J
Jan Kiszka 已提交
1605 1606 1607 1608 1609 1610
    for (i = 0; i < 4; i++) {
        qdev_connect_gpio_out(key_dev, i, qdev_get_gpio_in(dev, i + 8));
    }
    for (i = 4; i < 8; i++) {
        qdev_connect_gpio_out(key_dev, i, qdev_get_gpio_in(dev, i + 15));
    }
1611

1612 1613 1614 1615
    wm8750_dev = i2c_create_slave(i2c, "wm8750", MP_WM_ADDR);
    dev = qdev_create(NULL, "mv88w8618_audio");
    s = sysbus_from_qdev(dev);
    qdev_prop_set_ptr(dev, "wm8750", wm8750_dev);
M
Markus Armbruster 已提交
1616
    qdev_init_nofail(dev);
1617 1618 1619
    sysbus_mmio_map(s, 0, MP_AUDIO_BASE);
    sysbus_connect_irq(s, 0, pic[MP_AUDIO_IRQ]);

1620 1621 1622 1623
    musicpal_binfo.ram_size = MP_RAM_DEFAULT_SIZE;
    musicpal_binfo.kernel_filename = kernel_filename;
    musicpal_binfo.kernel_cmdline = kernel_cmdline;
    musicpal_binfo.initrd_filename = initrd_filename;
B
balrog 已提交
1624
    arm_load_kernel(env, &musicpal_binfo);
1625 1626
}

1627
static QEMUMachine musicpal_machine = {
1628 1629 1630
    .name = "musicpal",
    .desc = "Marvell 88w8618 / MusicPal (ARM926EJ-S)",
    .init = musicpal_init,
1631
};
P
Paul Brook 已提交
1632

1633 1634 1635 1636 1637 1638 1639
static void musicpal_machine_init(void)
{
    qemu_register_machine(&musicpal_machine);
}

machine_init(musicpal_machine_init);

P
Paul Brook 已提交
1640 1641
static void musicpal_register_devices(void)
{
J
Jan Kiszka 已提交
1642
    sysbus_register_withprop(&mv88w8618_pic_info);
1643
    sysbus_register_withprop(&mv88w8618_pit_info);
J
Jan Kiszka 已提交
1644 1645
    sysbus_register_withprop(&mv88w8618_flashcfg_info);
    sysbus_register_withprop(&mv88w8618_eth_info);
P
Paul Brook 已提交
1646 1647
    sysbus_register_dev("mv88w8618_wlan", sizeof(SysBusDevice),
                        mv88w8618_wlan_init);
J
Jan Kiszka 已提交
1648
    sysbus_register_withprop(&musicpal_lcd_info);
J
Jan Kiszka 已提交
1649
    sysbus_register_withprop(&musicpal_gpio_info);
J
Jan Kiszka 已提交
1650
    sysbus_register_withprop(&musicpal_key_info);
P
Paul Brook 已提交
1651 1652 1653
}

device_init(musicpal_register_devices)