2007-04-30 05:26:42 +04:00
|
|
|
/*
|
|
|
|
* Intel XScale PXA255/270 processor support.
|
|
|
|
*
|
|
|
|
* Copyright (c) 2006 Openedhand Ltd.
|
|
|
|
* Written by Andrzej Zaborowski <balrog@zabor.org>
|
|
|
|
*
|
2011-06-26 06:21:35 +04:00
|
|
|
* This code is licensed under the GPL.
|
2007-04-30 05:26:42 +04:00
|
|
|
*/
|
|
|
|
|
2009-05-15 01:35:09 +04:00
|
|
|
#include "sysbus.h"
|
2007-11-17 20:14:51 +03:00
|
|
|
#include "pxa.h"
|
|
|
|
#include "sysemu.h"
|
|
|
|
#include "pc.h"
|
|
|
|
#include "i2c.h"
|
2009-05-15 01:35:09 +04:00
|
|
|
#include "ssi.h"
|
2007-11-17 20:14:51 +03:00
|
|
|
#include "qemu-char.h"
|
2010-08-24 19:22:24 +04:00
|
|
|
#include "blockdev.h"
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
static struct {
|
2009-10-02 01:12:16 +04:00
|
|
|
target_phys_addr_t io_base;
|
2007-04-30 05:26:42 +04:00
|
|
|
int irqn;
|
|
|
|
} pxa255_serial[] = {
|
|
|
|
{ 0x40100000, PXA2XX_PIC_FFUART },
|
|
|
|
{ 0x40200000, PXA2XX_PIC_BTUART },
|
|
|
|
{ 0x40700000, PXA2XX_PIC_STUART },
|
|
|
|
{ 0x41600000, PXA25X_PIC_HWUART },
|
|
|
|
{ 0, 0 }
|
|
|
|
}, pxa270_serial[] = {
|
|
|
|
{ 0x40100000, PXA2XX_PIC_FFUART },
|
|
|
|
{ 0x40200000, PXA2XX_PIC_BTUART },
|
|
|
|
{ 0x40700000, PXA2XX_PIC_STUART },
|
|
|
|
{ 0, 0 }
|
|
|
|
};
|
|
|
|
|
2007-11-11 22:48:33 +03:00
|
|
|
typedef struct PXASSPDef {
|
2009-10-02 01:12:16 +04:00
|
|
|
target_phys_addr_t io_base;
|
2007-04-30 05:26:42 +04:00
|
|
|
int irqn;
|
2007-11-11 22:48:33 +03:00
|
|
|
} PXASSPDef;
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
static PXASSPDef pxa250_ssp[] = {
|
2007-04-30 05:26:42 +04:00
|
|
|
{ 0x41000000, PXA2XX_PIC_SSP },
|
|
|
|
{ 0, 0 }
|
2007-11-11 22:48:33 +03:00
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static PXASSPDef pxa255_ssp[] = {
|
2007-04-30 05:26:42 +04:00
|
|
|
{ 0x41000000, PXA2XX_PIC_SSP },
|
|
|
|
{ 0x41400000, PXA25X_PIC_NSSP },
|
|
|
|
{ 0, 0 }
|
2007-11-11 22:48:33 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
static PXASSPDef pxa26x_ssp[] = {
|
2007-04-30 05:26:42 +04:00
|
|
|
{ 0x41000000, PXA2XX_PIC_SSP },
|
|
|
|
{ 0x41400000, PXA25X_PIC_NSSP },
|
|
|
|
{ 0x41500000, PXA26X_PIC_ASSP },
|
|
|
|
{ 0, 0 }
|
2007-11-11 22:48:33 +03:00
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static PXASSPDef pxa27x_ssp[] = {
|
2007-04-30 05:26:42 +04:00
|
|
|
{ 0x41000000, PXA2XX_PIC_SSP },
|
|
|
|
{ 0x41700000, PXA27X_PIC_SSP2 },
|
|
|
|
{ 0x41900000, PXA2XX_PIC_SSP3 },
|
|
|
|
{ 0, 0 }
|
|
|
|
};
|
|
|
|
|
|
|
|
#define PMCR 0x00 /* Power Manager Control register */
|
|
|
|
#define PSSR 0x04 /* Power Manager Sleep Status register */
|
|
|
|
#define PSPR 0x08 /* Power Manager Scratch-Pad register */
|
|
|
|
#define PWER 0x0c /* Power Manager Wake-Up Enable register */
|
|
|
|
#define PRER 0x10 /* Power Manager Rising-Edge Detect Enable register */
|
|
|
|
#define PFER 0x14 /* Power Manager Falling-Edge Detect Enable register */
|
|
|
|
#define PEDR 0x18 /* Power Manager Edge-Detect Status register */
|
|
|
|
#define PCFR 0x1c /* Power Manager General Configuration register */
|
|
|
|
#define PGSR0 0x20 /* Power Manager GPIO Sleep-State register 0 */
|
|
|
|
#define PGSR1 0x24 /* Power Manager GPIO Sleep-State register 1 */
|
|
|
|
#define PGSR2 0x28 /* Power Manager GPIO Sleep-State register 2 */
|
|
|
|
#define PGSR3 0x2c /* Power Manager GPIO Sleep-State register 3 */
|
|
|
|
#define RCSR 0x30 /* Reset Controller Status register */
|
|
|
|
#define PSLR 0x34 /* Power Manager Sleep Configuration register */
|
|
|
|
#define PTSR 0x38 /* Power Manager Standby Configuration register */
|
|
|
|
#define PVCR 0x40 /* Power Manager Voltage Change Control register */
|
|
|
|
#define PUCR 0x4c /* Power Manager USIM Card Control/Status register */
|
|
|
|
#define PKWR 0x50 /* Power Manager Keyboard Wake-Up Enable register */
|
|
|
|
#define PKSR 0x54 /* Power Manager Keyboard Level-Detect Status */
|
|
|
|
#define PCMD0 0x80 /* Power Manager I2C Command register File 0 */
|
|
|
|
#define PCMD31 0xfc /* Power Manager I2C Command register File 31 */
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static uint64_t pxa2xx_pm_read(void *opaque, target_phys_addr_t addr,
|
|
|
|
unsigned size)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxState *s = (PXA2xxState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
switch (addr) {
|
|
|
|
case PMCR ... PCMD31:
|
|
|
|
if (addr & 3)
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
return s->pm_regs[addr >> 2];
|
|
|
|
default:
|
|
|
|
fail:
|
|
|
|
printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-10-02 01:12:16 +04:00
|
|
|
static void pxa2xx_pm_write(void *opaque, target_phys_addr_t addr,
|
2011-09-25 19:19:19 +04:00
|
|
|
uint64_t value, unsigned size)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxState *s = (PXA2xxState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
switch (addr) {
|
|
|
|
case PMCR:
|
2011-11-10 00:46:35 +04:00
|
|
|
/* Clear the write-one-to-clear bits... */
|
|
|
|
s->pm_regs[addr >> 2] &= ~(value & 0x2a);
|
|
|
|
/* ...and set the plain r/w bits */
|
2011-11-13 18:18:39 +04:00
|
|
|
s->pm_regs[addr >> 2] &= ~0x15;
|
2007-04-30 05:26:42 +04:00
|
|
|
s->pm_regs[addr >> 2] |= value & 0x15;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PSSR: /* Read-clean registers */
|
|
|
|
case RCSR:
|
|
|
|
case PKSR:
|
|
|
|
s->pm_regs[addr >> 2] &= ~value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default: /* Read-write registers */
|
2010-09-18 09:53:15 +04:00
|
|
|
if (!(addr & 3)) {
|
2007-04-30 05:26:42 +04:00
|
|
|
s->pm_regs[addr >> 2] = value;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static const MemoryRegionOps pxa2xx_pm_ops = {
|
|
|
|
.read = pxa2xx_pm_read,
|
|
|
|
.write = pxa2xx_pm_write,
|
|
|
|
.endianness = DEVICE_NATIVE_ENDIAN,
|
2007-04-30 05:26:42 +04:00
|
|
|
};
|
|
|
|
|
2010-12-02 16:54:38 +03:00
|
|
|
static const VMStateDescription vmstate_pxa2xx_pm = {
|
|
|
|
.name = "pxa2xx_pm",
|
|
|
|
.version_id = 0,
|
|
|
|
.minimum_version_id = 0,
|
|
|
|
.minimum_version_id_old = 0,
|
|
|
|
.fields = (VMStateField[]) {
|
|
|
|
VMSTATE_UINT32_ARRAY(pm_regs, PXA2xxState, 0x40),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
2007-05-24 22:50:09 +04:00
|
|
|
|
2007-04-30 05:26:42 +04:00
|
|
|
#define CCCR 0x00 /* Core Clock Configuration register */
|
|
|
|
#define CKEN 0x04 /* Clock Enable register */
|
|
|
|
#define OSCC 0x08 /* Oscillator Configuration register */
|
|
|
|
#define CCSR 0x0c /* Core Clock Status register */
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static uint64_t pxa2xx_cm_read(void *opaque, target_phys_addr_t addr,
|
|
|
|
unsigned size)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxState *s = (PXA2xxState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
switch (addr) {
|
|
|
|
case CCCR:
|
|
|
|
case CKEN:
|
|
|
|
case OSCC:
|
|
|
|
return s->cm_regs[addr >> 2];
|
|
|
|
|
|
|
|
case CCSR:
|
|
|
|
return s->cm_regs[CCCR >> 2] | (3 << 28);
|
|
|
|
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-10-02 01:12:16 +04:00
|
|
|
static void pxa2xx_cm_write(void *opaque, target_phys_addr_t addr,
|
2011-09-25 19:19:19 +04:00
|
|
|
uint64_t value, unsigned size)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxState *s = (PXA2xxState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
switch (addr) {
|
|
|
|
case CCCR:
|
|
|
|
case CKEN:
|
|
|
|
s->cm_regs[addr >> 2] = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OSCC:
|
2007-05-08 23:20:04 +04:00
|
|
|
s->cm_regs[addr >> 2] &= ~0x6c;
|
2007-04-30 05:26:42 +04:00
|
|
|
s->cm_regs[addr >> 2] |= value & 0x6e;
|
2007-05-08 23:20:04 +04:00
|
|
|
if ((value >> 1) & 1) /* OON */
|
|
|
|
s->cm_regs[addr >> 2] |= 1 << 0; /* Oscillator is now stable */
|
2007-04-30 05:26:42 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static const MemoryRegionOps pxa2xx_cm_ops = {
|
|
|
|
.read = pxa2xx_cm_read,
|
|
|
|
.write = pxa2xx_cm_write,
|
|
|
|
.endianness = DEVICE_NATIVE_ENDIAN,
|
2007-04-30 05:26:42 +04:00
|
|
|
};
|
|
|
|
|
2010-12-02 16:31:14 +03:00
|
|
|
static const VMStateDescription vmstate_pxa2xx_cm = {
|
|
|
|
.name = "pxa2xx_cm",
|
|
|
|
.version_id = 0,
|
|
|
|
.minimum_version_id = 0,
|
|
|
|
.minimum_version_id_old = 0,
|
|
|
|
.fields = (VMStateField[]) {
|
|
|
|
VMSTATE_UINT32_ARRAY(cm_regs, PXA2xxState, 4),
|
|
|
|
VMSTATE_UINT32(clkcfg, PXA2xxState),
|
|
|
|
VMSTATE_UINT32(pmnc, PXA2xxState),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
2007-05-24 22:50:09 +04:00
|
|
|
|
2007-04-30 05:26:42 +04:00
|
|
|
static uint32_t pxa2xx_clkpwr_read(void *opaque, int op2, int reg, int crm)
|
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxState *s = (PXA2xxState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
switch (reg) {
|
|
|
|
case 6: /* Clock Configuration register */
|
|
|
|
return s->clkcfg;
|
|
|
|
|
|
|
|
case 7: /* Power Mode register */
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register 0x%x\n", __FUNCTION__, reg);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void pxa2xx_clkpwr_write(void *opaque, int op2, int reg, int crm,
|
|
|
|
uint32_t value)
|
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxState *s = (PXA2xxState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
static const char *pwrmode[8] = {
|
|
|
|
"Normal", "Idle", "Deep-idle", "Standby",
|
|
|
|
"Sleep", "reserved (!)", "reserved (!)", "Deep-sleep",
|
|
|
|
};
|
|
|
|
|
|
|
|
switch (reg) {
|
|
|
|
case 6: /* Clock Configuration register */
|
|
|
|
s->clkcfg = value & 0xf;
|
|
|
|
if (value & 2)
|
|
|
|
printf("%s: CPU frequency change attempt\n", __FUNCTION__);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 7: /* Power Mode register */
|
|
|
|
if (value & 8)
|
|
|
|
printf("%s: CPU voltage change attempt\n", __FUNCTION__);
|
|
|
|
switch (value & 7) {
|
|
|
|
case 0:
|
|
|
|
/* Do nothing */
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 1:
|
|
|
|
/* Idle */
|
2007-07-24 05:07:44 +04:00
|
|
|
if (!(s->cm_regs[CCCR >> 2] & (1 << 31))) { /* CPDIS */
|
2012-05-04 01:47:04 +04:00
|
|
|
cpu_interrupt(&s->cpu->env, CPU_INTERRUPT_HALT);
|
2007-04-30 05:26:42 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* Fall through. */
|
|
|
|
|
|
|
|
case 2:
|
|
|
|
/* Deep-Idle */
|
2012-05-04 01:47:04 +04:00
|
|
|
cpu_interrupt(&s->cpu->env, CPU_INTERRUPT_HALT);
|
2007-04-30 05:26:42 +04:00
|
|
|
s->pm_regs[RCSR >> 2] |= 0x8; /* Set GPR */
|
|
|
|
goto message;
|
|
|
|
|
|
|
|
case 3:
|
2012-05-04 01:47:04 +04:00
|
|
|
s->cpu->env.uncached_cpsr =
|
2007-05-01 05:28:01 +04:00
|
|
|
ARM_CPU_MODE_SVC | CPSR_A | CPSR_F | CPSR_I;
|
2012-05-04 01:47:04 +04:00
|
|
|
s->cpu->env.cp15.c1_sys = 0;
|
|
|
|
s->cpu->env.cp15.c1_coproc = 0;
|
|
|
|
s->cpu->env.cp15.c2_base0 = 0;
|
|
|
|
s->cpu->env.cp15.c3 = 0;
|
2007-04-30 05:26:42 +04:00
|
|
|
s->pm_regs[PSSR >> 2] |= 0x8; /* Set STS */
|
|
|
|
s->pm_regs[RCSR >> 2] |= 0x8; /* Set GPR */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The scratch-pad register is almost universally used
|
|
|
|
* for storing the return address on suspend. For the
|
|
|
|
* lack of a resuming bootloader, perform a jump
|
|
|
|
* directly to that address.
|
|
|
|
*/
|
2012-05-04 01:47:04 +04:00
|
|
|
memset(s->cpu->env.regs, 0, 4 * 15);
|
|
|
|
s->cpu->env.regs[15] = s->pm_regs[PSPR >> 2];
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
#if 0
|
|
|
|
buffer = 0xe59ff000; /* ldr pc, [pc, #0] */
|
|
|
|
cpu_physical_memory_write(0, &buffer, 4);
|
|
|
|
buffer = s->pm_regs[PSPR >> 2];
|
|
|
|
cpu_physical_memory_write(8, &buffer, 4);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Suspend */
|
|
|
|
cpu_interrupt(cpu_single_env, CPU_INTERRUPT_HALT);
|
|
|
|
|
|
|
|
goto message;
|
|
|
|
|
|
|
|
default:
|
|
|
|
message:
|
|
|
|
printf("%s: machine entered %s mode\n", __FUNCTION__,
|
|
|
|
pwrmode[value & 7]);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register 0x%x\n", __FUNCTION__, reg);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Performace Monitoring Registers */
|
|
|
|
#define CPPMNC 0 /* Performance Monitor Control register */
|
|
|
|
#define CPCCNT 1 /* Clock Counter register */
|
|
|
|
#define CPINTEN 4 /* Interrupt Enable register */
|
|
|
|
#define CPFLAG 5 /* Overflow Flag register */
|
|
|
|
#define CPEVTSEL 8 /* Event Selection register */
|
|
|
|
|
|
|
|
#define CPPMN0 0 /* Performance Count register 0 */
|
|
|
|
#define CPPMN1 1 /* Performance Count register 1 */
|
|
|
|
#define CPPMN2 2 /* Performance Count register 2 */
|
|
|
|
#define CPPMN3 3 /* Performance Count register 3 */
|
|
|
|
|
|
|
|
static uint32_t pxa2xx_perf_read(void *opaque, int op2, int reg, int crm)
|
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxState *s = (PXA2xxState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
switch (reg) {
|
|
|
|
case CPPMNC:
|
|
|
|
return s->pmnc;
|
|
|
|
case CPCCNT:
|
|
|
|
if (s->pmnc & 1)
|
2011-03-11 18:47:48 +03:00
|
|
|
return qemu_get_clock_ns(vm_clock);
|
2007-04-30 05:26:42 +04:00
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
case CPINTEN:
|
|
|
|
case CPFLAG:
|
|
|
|
case CPEVTSEL:
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register 0x%x\n", __FUNCTION__, reg);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void pxa2xx_perf_write(void *opaque, int op2, int reg, int crm,
|
|
|
|
uint32_t value)
|
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxState *s = (PXA2xxState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
switch (reg) {
|
|
|
|
case CPPMNC:
|
|
|
|
s->pmnc = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CPCCNT:
|
|
|
|
case CPINTEN:
|
|
|
|
case CPFLAG:
|
|
|
|
case CPEVTSEL:
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register 0x%x\n", __FUNCTION__, reg);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint32_t pxa2xx_cp14_read(void *opaque, int op2, int reg, int crm)
|
|
|
|
{
|
|
|
|
switch (crm) {
|
|
|
|
case 0:
|
|
|
|
return pxa2xx_clkpwr_read(opaque, op2, reg, crm);
|
|
|
|
case 1:
|
|
|
|
return pxa2xx_perf_read(opaque, op2, reg, crm);
|
|
|
|
case 2:
|
|
|
|
switch (reg) {
|
|
|
|
case CPPMN0:
|
|
|
|
case CPPMN1:
|
|
|
|
case CPPMN2:
|
|
|
|
case CPPMN3:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
/* Fall through */
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register 0x%x\n", __FUNCTION__, reg);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void pxa2xx_cp14_write(void *opaque, int op2, int reg, int crm,
|
|
|
|
uint32_t value)
|
|
|
|
{
|
|
|
|
switch (crm) {
|
|
|
|
case 0:
|
|
|
|
pxa2xx_clkpwr_write(opaque, op2, reg, crm, value);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
pxa2xx_perf_write(opaque, op2, reg, crm, value);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
switch (reg) {
|
|
|
|
case CPPMN0:
|
|
|
|
case CPPMN1:
|
|
|
|
case CPPMN2:
|
|
|
|
case CPPMN3:
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
/* Fall through */
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register 0x%x\n", __FUNCTION__, reg);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#define MDCNFG 0x00 /* SDRAM Configuration register */
|
|
|
|
#define MDREFR 0x04 /* SDRAM Refresh Control register */
|
|
|
|
#define MSC0 0x08 /* Static Memory Control register 0 */
|
|
|
|
#define MSC1 0x0c /* Static Memory Control register 1 */
|
|
|
|
#define MSC2 0x10 /* Static Memory Control register 2 */
|
|
|
|
#define MECR 0x14 /* Expansion Memory Bus Config register */
|
|
|
|
#define SXCNFG 0x1c /* Synchronous Static Memory Config register */
|
|
|
|
#define MCMEM0 0x28 /* PC Card Memory Socket 0 Timing register */
|
|
|
|
#define MCMEM1 0x2c /* PC Card Memory Socket 1 Timing register */
|
|
|
|
#define MCATT0 0x30 /* PC Card Attribute Socket 0 register */
|
|
|
|
#define MCATT1 0x34 /* PC Card Attribute Socket 1 register */
|
|
|
|
#define MCIO0 0x38 /* PC Card I/O Socket 0 Timing register */
|
|
|
|
#define MCIO1 0x3c /* PC Card I/O Socket 1 Timing register */
|
|
|
|
#define MDMRS 0x40 /* SDRAM Mode Register Set Config register */
|
|
|
|
#define BOOT_DEF 0x44 /* Boot-time Default Configuration register */
|
|
|
|
#define ARB_CNTL 0x48 /* Arbiter Control register */
|
|
|
|
#define BSCNTR0 0x4c /* Memory Buffer Strength Control register 0 */
|
|
|
|
#define BSCNTR1 0x50 /* Memory Buffer Strength Control register 1 */
|
|
|
|
#define LCDBSCNTR 0x54 /* LCD Buffer Strength Control register */
|
|
|
|
#define MDMRSLP 0x58 /* Low Power SDRAM Mode Set Config register */
|
|
|
|
#define BSCNTR2 0x5c /* Memory Buffer Strength Control register 2 */
|
|
|
|
#define BSCNTR3 0x60 /* Memory Buffer Strength Control register 3 */
|
|
|
|
#define SA1110 0x64 /* SA-1110 Memory Compatibility register */
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static uint64_t pxa2xx_mm_read(void *opaque, target_phys_addr_t addr,
|
|
|
|
unsigned size)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxState *s = (PXA2xxState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
switch (addr) {
|
|
|
|
case MDCNFG ... SA1110:
|
|
|
|
if ((addr & 3) == 0)
|
|
|
|
return s->mm_regs[addr >> 2];
|
|
|
|
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-10-02 01:12:16 +04:00
|
|
|
static void pxa2xx_mm_write(void *opaque, target_phys_addr_t addr,
|
2011-09-25 19:19:19 +04:00
|
|
|
uint64_t value, unsigned size)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxState *s = (PXA2xxState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
switch (addr) {
|
|
|
|
case MDCNFG ... SA1110:
|
|
|
|
if ((addr & 3) == 0) {
|
|
|
|
s->mm_regs[addr >> 2] = value;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static const MemoryRegionOps pxa2xx_mm_ops = {
|
|
|
|
.read = pxa2xx_mm_read,
|
|
|
|
.write = pxa2xx_mm_write,
|
|
|
|
.endianness = DEVICE_NATIVE_ENDIAN,
|
2007-04-30 05:26:42 +04:00
|
|
|
};
|
|
|
|
|
2010-12-02 16:36:57 +03:00
|
|
|
static const VMStateDescription vmstate_pxa2xx_mm = {
|
|
|
|
.name = "pxa2xx_mm",
|
|
|
|
.version_id = 0,
|
|
|
|
.minimum_version_id = 0,
|
|
|
|
.minimum_version_id_old = 0,
|
|
|
|
.fields = (VMStateField[]) {
|
|
|
|
VMSTATE_UINT32_ARRAY(mm_regs, PXA2xxState, 0x1a),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
2007-05-24 22:50:09 +04:00
|
|
|
|
2007-04-30 05:26:42 +04:00
|
|
|
/* Synchronous Serial Ports */
|
2009-05-15 01:35:09 +04:00
|
|
|
typedef struct {
|
|
|
|
SysBusDevice busdev;
|
2011-09-25 19:19:19 +04:00
|
|
|
MemoryRegion iomem;
|
2007-04-30 05:26:42 +04:00
|
|
|
qemu_irq irq;
|
|
|
|
int enable;
|
2009-05-15 01:35:09 +04:00
|
|
|
SSIBus *bus;
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
uint32_t sscr[2];
|
|
|
|
uint32_t sspsp;
|
|
|
|
uint32_t ssto;
|
|
|
|
uint32_t ssitr;
|
|
|
|
uint32_t sssr;
|
|
|
|
uint8_t sstsa;
|
|
|
|
uint8_t ssrsa;
|
|
|
|
uint8_t ssacd;
|
|
|
|
|
|
|
|
uint32_t rx_fifo[16];
|
|
|
|
int rx_level;
|
|
|
|
int rx_start;
|
2009-05-15 01:35:09 +04:00
|
|
|
} PXA2xxSSPState;
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
#define SSCR0 0x00 /* SSP Control register 0 */
|
|
|
|
#define SSCR1 0x04 /* SSP Control register 1 */
|
|
|
|
#define SSSR 0x08 /* SSP Status register */
|
|
|
|
#define SSITR 0x0c /* SSP Interrupt Test register */
|
|
|
|
#define SSDR 0x10 /* SSP Data register */
|
|
|
|
#define SSTO 0x28 /* SSP Time-Out register */
|
|
|
|
#define SSPSP 0x2c /* SSP Programmable Serial Protocol register */
|
|
|
|
#define SSTSA 0x30 /* SSP TX Time Slot Active register */
|
|
|
|
#define SSRSA 0x34 /* SSP RX Time Slot Active register */
|
|
|
|
#define SSTSS 0x38 /* SSP Time Slot Status register */
|
|
|
|
#define SSACD 0x3c /* SSP Audio Clock Divider register */
|
|
|
|
|
|
|
|
/* Bitfields for above registers */
|
|
|
|
#define SSCR0_SPI(x) (((x) & 0x30) == 0x00)
|
|
|
|
#define SSCR0_SSP(x) (((x) & 0x30) == 0x10)
|
|
|
|
#define SSCR0_UWIRE(x) (((x) & 0x30) == 0x20)
|
|
|
|
#define SSCR0_PSP(x) (((x) & 0x30) == 0x30)
|
|
|
|
#define SSCR0_SSE (1 << 7)
|
|
|
|
#define SSCR0_RIM (1 << 22)
|
|
|
|
#define SSCR0_TIM (1 << 23)
|
|
|
|
#define SSCR0_MOD (1 << 31)
|
|
|
|
#define SSCR0_DSS(x) (((((x) >> 16) & 0x10) | ((x) & 0xf)) + 1)
|
|
|
|
#define SSCR1_RIE (1 << 0)
|
|
|
|
#define SSCR1_TIE (1 << 1)
|
|
|
|
#define SSCR1_LBM (1 << 2)
|
|
|
|
#define SSCR1_MWDS (1 << 5)
|
|
|
|
#define SSCR1_TFT(x) ((((x) >> 6) & 0xf) + 1)
|
|
|
|
#define SSCR1_RFT(x) ((((x) >> 10) & 0xf) + 1)
|
|
|
|
#define SSCR1_EFWR (1 << 14)
|
|
|
|
#define SSCR1_PINTE (1 << 18)
|
|
|
|
#define SSCR1_TINTE (1 << 19)
|
|
|
|
#define SSCR1_RSRE (1 << 20)
|
|
|
|
#define SSCR1_TSRE (1 << 21)
|
|
|
|
#define SSCR1_EBCEI (1 << 29)
|
|
|
|
#define SSITR_INT (7 << 5)
|
|
|
|
#define SSSR_TNF (1 << 2)
|
|
|
|
#define SSSR_RNE (1 << 3)
|
|
|
|
#define SSSR_TFS (1 << 5)
|
|
|
|
#define SSSR_RFS (1 << 6)
|
|
|
|
#define SSSR_ROR (1 << 7)
|
|
|
|
#define SSSR_PINT (1 << 18)
|
|
|
|
#define SSSR_TINT (1 << 19)
|
|
|
|
#define SSSR_EOC (1 << 20)
|
|
|
|
#define SSSR_TUR (1 << 21)
|
|
|
|
#define SSSR_BCE (1 << 23)
|
|
|
|
#define SSSR_RW 0x00bc0080
|
|
|
|
|
2009-05-10 04:44:56 +04:00
|
|
|
static void pxa2xx_ssp_int_update(PXA2xxSSPState *s)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
|
|
|
int level = 0;
|
|
|
|
|
|
|
|
level |= s->ssitr & SSITR_INT;
|
|
|
|
level |= (s->sssr & SSSR_BCE) && (s->sscr[1] & SSCR1_EBCEI);
|
|
|
|
level |= (s->sssr & SSSR_TUR) && !(s->sscr[0] & SSCR0_TIM);
|
|
|
|
level |= (s->sssr & SSSR_EOC) && (s->sssr & (SSSR_TINT | SSSR_PINT));
|
|
|
|
level |= (s->sssr & SSSR_TINT) && (s->sscr[1] & SSCR1_TINTE);
|
|
|
|
level |= (s->sssr & SSSR_PINT) && (s->sscr[1] & SSCR1_PINTE);
|
|
|
|
level |= (s->sssr & SSSR_ROR) && !(s->sscr[0] & SSCR0_RIM);
|
|
|
|
level |= (s->sssr & SSSR_RFS) && (s->sscr[1] & SSCR1_RIE);
|
|
|
|
level |= (s->sssr & SSSR_TFS) && (s->sscr[1] & SSCR1_TIE);
|
|
|
|
qemu_set_irq(s->irq, !!level);
|
|
|
|
}
|
|
|
|
|
2009-05-10 04:44:56 +04:00
|
|
|
static void pxa2xx_ssp_fifo_update(PXA2xxSSPState *s)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
|
|
|
s->sssr &= ~(0xf << 12); /* Clear RFL */
|
|
|
|
s->sssr &= ~(0xf << 8); /* Clear TFL */
|
2010-09-18 09:53:15 +04:00
|
|
|
s->sssr &= ~SSSR_TFS;
|
2007-04-30 05:26:42 +04:00
|
|
|
s->sssr &= ~SSSR_TNF;
|
|
|
|
if (s->enable) {
|
|
|
|
s->sssr |= ((s->rx_level - 1) & 0xf) << 12;
|
|
|
|
if (s->rx_level >= SSCR1_RFT(s->sscr[1]))
|
|
|
|
s->sssr |= SSSR_RFS;
|
|
|
|
else
|
|
|
|
s->sssr &= ~SSSR_RFS;
|
|
|
|
if (s->rx_level)
|
|
|
|
s->sssr |= SSSR_RNE;
|
|
|
|
else
|
|
|
|
s->sssr &= ~SSSR_RNE;
|
2010-09-18 09:53:15 +04:00
|
|
|
/* TX FIFO is never filled, so it is always in underrun
|
|
|
|
condition if SSP is enabled */
|
|
|
|
s->sssr |= SSSR_TFS;
|
2007-04-30 05:26:42 +04:00
|
|
|
s->sssr |= SSSR_TNF;
|
|
|
|
}
|
|
|
|
|
|
|
|
pxa2xx_ssp_int_update(s);
|
|
|
|
}
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static uint64_t pxa2xx_ssp_read(void *opaque, target_phys_addr_t addr,
|
|
|
|
unsigned size)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxSSPState *s = (PXA2xxSSPState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
uint32_t retval;
|
|
|
|
|
|
|
|
switch (addr) {
|
|
|
|
case SSCR0:
|
|
|
|
return s->sscr[0];
|
|
|
|
case SSCR1:
|
|
|
|
return s->sscr[1];
|
|
|
|
case SSPSP:
|
|
|
|
return s->sspsp;
|
|
|
|
case SSTO:
|
|
|
|
return s->ssto;
|
|
|
|
case SSITR:
|
|
|
|
return s->ssitr;
|
|
|
|
case SSSR:
|
|
|
|
return s->sssr | s->ssitr;
|
|
|
|
case SSDR:
|
|
|
|
if (!s->enable)
|
|
|
|
return 0xffffffff;
|
|
|
|
if (s->rx_level < 1) {
|
|
|
|
printf("%s: SSP Rx Underrun\n", __FUNCTION__);
|
|
|
|
return 0xffffffff;
|
|
|
|
}
|
|
|
|
s->rx_level --;
|
|
|
|
retval = s->rx_fifo[s->rx_start ++];
|
|
|
|
s->rx_start &= 0xf;
|
|
|
|
pxa2xx_ssp_fifo_update(s);
|
|
|
|
return retval;
|
|
|
|
case SSTSA:
|
|
|
|
return s->sstsa;
|
|
|
|
case SSRSA:
|
|
|
|
return s->ssrsa;
|
|
|
|
case SSTSS:
|
|
|
|
return 0;
|
|
|
|
case SSACD:
|
|
|
|
return s->ssacd;
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-10-02 01:12:16 +04:00
|
|
|
static void pxa2xx_ssp_write(void *opaque, target_phys_addr_t addr,
|
2011-09-25 19:19:19 +04:00
|
|
|
uint64_t value64, unsigned size)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxSSPState *s = (PXA2xxSSPState *) opaque;
|
2011-09-25 19:19:19 +04:00
|
|
|
uint32_t value = value64;
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
switch (addr) {
|
|
|
|
case SSCR0:
|
|
|
|
s->sscr[0] = value & 0xc7ffffff;
|
|
|
|
s->enable = value & SSCR0_SSE;
|
|
|
|
if (value & SSCR0_MOD)
|
|
|
|
printf("%s: Attempt to use network mode\n", __FUNCTION__);
|
|
|
|
if (s->enable && SSCR0_DSS(value) < 4)
|
|
|
|
printf("%s: Wrong data size: %i bits\n", __FUNCTION__,
|
|
|
|
SSCR0_DSS(value));
|
|
|
|
if (!(value & SSCR0_SSE)) {
|
|
|
|
s->sssr = 0;
|
|
|
|
s->ssitr = 0;
|
|
|
|
s->rx_level = 0;
|
|
|
|
}
|
|
|
|
pxa2xx_ssp_fifo_update(s);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SSCR1:
|
|
|
|
s->sscr[1] = value;
|
|
|
|
if (value & (SSCR1_LBM | SSCR1_EFWR))
|
|
|
|
printf("%s: Attempt to use SSP test mode\n", __FUNCTION__);
|
|
|
|
pxa2xx_ssp_fifo_update(s);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SSPSP:
|
|
|
|
s->sspsp = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SSTO:
|
|
|
|
s->ssto = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SSITR:
|
|
|
|
s->ssitr = value & SSITR_INT;
|
|
|
|
pxa2xx_ssp_int_update(s);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SSSR:
|
|
|
|
s->sssr &= ~(value & SSSR_RW);
|
|
|
|
pxa2xx_ssp_int_update(s);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SSDR:
|
|
|
|
if (SSCR0_UWIRE(s->sscr[0])) {
|
|
|
|
if (s->sscr[1] & SSCR1_MWDS)
|
|
|
|
value &= 0xffff;
|
|
|
|
else
|
|
|
|
value &= 0xff;
|
|
|
|
} else
|
|
|
|
/* Note how 32bits overflow does no harm here */
|
|
|
|
value &= (1 << SSCR0_DSS(s->sscr[0])) - 1;
|
|
|
|
|
|
|
|
/* Data goes from here to the Tx FIFO and is shifted out from
|
|
|
|
* there directly to the slave, no need to buffer it.
|
|
|
|
*/
|
|
|
|
if (s->enable) {
|
2009-05-15 01:35:09 +04:00
|
|
|
uint32_t readval;
|
|
|
|
readval = ssi_transfer(s->bus, value);
|
2007-04-30 05:26:42 +04:00
|
|
|
if (s->rx_level < 0x10) {
|
2009-05-15 01:35:09 +04:00
|
|
|
s->rx_fifo[(s->rx_start + s->rx_level ++) & 0xf] = readval;
|
|
|
|
} else {
|
2007-04-30 05:26:42 +04:00
|
|
|
s->sssr |= SSSR_ROR;
|
2009-05-15 01:35:09 +04:00
|
|
|
}
|
2007-04-30 05:26:42 +04:00
|
|
|
}
|
|
|
|
pxa2xx_ssp_fifo_update(s);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SSTSA:
|
|
|
|
s->sstsa = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SSRSA:
|
|
|
|
s->ssrsa = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SSACD:
|
|
|
|
s->ssacd = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static const MemoryRegionOps pxa2xx_ssp_ops = {
|
|
|
|
.read = pxa2xx_ssp_read,
|
|
|
|
.write = pxa2xx_ssp_write,
|
|
|
|
.endianness = DEVICE_NATIVE_ENDIAN,
|
2007-04-30 05:26:42 +04:00
|
|
|
};
|
|
|
|
|
2007-05-24 22:50:09 +04:00
|
|
|
static void pxa2xx_ssp_save(QEMUFile *f, void *opaque)
|
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxSSPState *s = (PXA2xxSSPState *) opaque;
|
2007-05-24 22:50:09 +04:00
|
|
|
int i;
|
|
|
|
|
|
|
|
qemu_put_be32(f, s->enable);
|
|
|
|
|
|
|
|
qemu_put_be32s(f, &s->sscr[0]);
|
|
|
|
qemu_put_be32s(f, &s->sscr[1]);
|
|
|
|
qemu_put_be32s(f, &s->sspsp);
|
|
|
|
qemu_put_be32s(f, &s->ssto);
|
|
|
|
qemu_put_be32s(f, &s->ssitr);
|
|
|
|
qemu_put_be32s(f, &s->sssr);
|
|
|
|
qemu_put_8s(f, &s->sstsa);
|
|
|
|
qemu_put_8s(f, &s->ssrsa);
|
|
|
|
qemu_put_8s(f, &s->ssacd);
|
|
|
|
|
|
|
|
qemu_put_byte(f, s->rx_level);
|
|
|
|
for (i = 0; i < s->rx_level; i ++)
|
|
|
|
qemu_put_byte(f, s->rx_fifo[(s->rx_start + i) & 0xf]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int pxa2xx_ssp_load(QEMUFile *f, void *opaque, int version_id)
|
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxSSPState *s = (PXA2xxSSPState *) opaque;
|
2007-05-24 22:50:09 +04:00
|
|
|
int i;
|
|
|
|
|
|
|
|
s->enable = qemu_get_be32(f);
|
|
|
|
|
|
|
|
qemu_get_be32s(f, &s->sscr[0]);
|
|
|
|
qemu_get_be32s(f, &s->sscr[1]);
|
|
|
|
qemu_get_be32s(f, &s->sspsp);
|
|
|
|
qemu_get_be32s(f, &s->ssto);
|
|
|
|
qemu_get_be32s(f, &s->ssitr);
|
|
|
|
qemu_get_be32s(f, &s->sssr);
|
|
|
|
qemu_get_8s(f, &s->sstsa);
|
|
|
|
qemu_get_8s(f, &s->ssrsa);
|
|
|
|
qemu_get_8s(f, &s->ssacd);
|
|
|
|
|
|
|
|
s->rx_level = qemu_get_byte(f);
|
|
|
|
s->rx_start = 0;
|
|
|
|
for (i = 0; i < s->rx_level; i ++)
|
|
|
|
s->rx_fifo[i] = qemu_get_byte(f);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-08-14 12:36:05 +04:00
|
|
|
static int pxa2xx_ssp_init(SysBusDevice *dev)
|
2009-05-15 01:35:09 +04:00
|
|
|
{
|
|
|
|
PXA2xxSSPState *s = FROM_SYSBUS(PXA2xxSSPState, dev);
|
|
|
|
|
|
|
|
sysbus_init_irq(dev, &s->irq);
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
memory_region_init_io(&s->iomem, &pxa2xx_ssp_ops, s, "pxa2xx-ssp", 0x1000);
|
2011-11-27 13:38:10 +04:00
|
|
|
sysbus_init_mmio(dev, &s->iomem);
|
2010-06-25 21:09:07 +04:00
|
|
|
register_savevm(&dev->qdev, "pxa2xx_ssp", -1, 0,
|
2009-05-15 01:35:09 +04:00
|
|
|
pxa2xx_ssp_save, pxa2xx_ssp_load, s);
|
|
|
|
|
2009-05-23 03:05:19 +04:00
|
|
|
s->bus = ssi_create_bus(&dev->qdev, "ssi");
|
2009-08-14 12:36:05 +04:00
|
|
|
return 0;
|
2009-05-15 01:35:09 +04:00
|
|
|
}
|
|
|
|
|
2007-04-30 05:26:42 +04:00
|
|
|
/* Real-Time Clock */
|
|
|
|
#define RCNR 0x00 /* RTC Counter register */
|
|
|
|
#define RTAR 0x04 /* RTC Alarm register */
|
|
|
|
#define RTSR 0x08 /* RTC Status register */
|
|
|
|
#define RTTR 0x0c /* RTC Timer Trim register */
|
|
|
|
#define RDCR 0x10 /* RTC Day Counter register */
|
|
|
|
#define RYCR 0x14 /* RTC Year Counter register */
|
|
|
|
#define RDAR1 0x18 /* RTC Wristwatch Day Alarm register 1 */
|
|
|
|
#define RYAR1 0x1c /* RTC Wristwatch Year Alarm register 1 */
|
|
|
|
#define RDAR2 0x20 /* RTC Wristwatch Day Alarm register 2 */
|
|
|
|
#define RYAR2 0x24 /* RTC Wristwatch Year Alarm register 2 */
|
|
|
|
#define SWCR 0x28 /* RTC Stopwatch Counter register */
|
|
|
|
#define SWAR1 0x2c /* RTC Stopwatch Alarm register 1 */
|
|
|
|
#define SWAR2 0x30 /* RTC Stopwatch Alarm register 2 */
|
|
|
|
#define RTCPICR 0x34 /* RTC Periodic Interrupt Counter register */
|
|
|
|
#define PIAR 0x38 /* RTC Periodic Interrupt Alarm register */
|
|
|
|
|
2011-03-03 17:13:42 +03:00
|
|
|
typedef struct {
|
|
|
|
SysBusDevice busdev;
|
2011-09-25 19:19:19 +04:00
|
|
|
MemoryRegion iomem;
|
2011-03-03 17:13:42 +03:00
|
|
|
uint32_t rttr;
|
|
|
|
uint32_t rtsr;
|
|
|
|
uint32_t rtar;
|
|
|
|
uint32_t rdar1;
|
|
|
|
uint32_t rdar2;
|
|
|
|
uint32_t ryar1;
|
|
|
|
uint32_t ryar2;
|
|
|
|
uint32_t swar1;
|
|
|
|
uint32_t swar2;
|
|
|
|
uint32_t piar;
|
|
|
|
uint32_t last_rcnr;
|
|
|
|
uint32_t last_rdcr;
|
|
|
|
uint32_t last_rycr;
|
|
|
|
uint32_t last_swcr;
|
|
|
|
uint32_t last_rtcpicr;
|
|
|
|
int64_t last_hz;
|
|
|
|
int64_t last_sw;
|
|
|
|
int64_t last_pi;
|
|
|
|
QEMUTimer *rtc_hz;
|
|
|
|
QEMUTimer *rtc_rdal1;
|
|
|
|
QEMUTimer *rtc_rdal2;
|
|
|
|
QEMUTimer *rtc_swal1;
|
|
|
|
QEMUTimer *rtc_swal2;
|
|
|
|
QEMUTimer *rtc_pi;
|
|
|
|
qemu_irq rtc_irq;
|
|
|
|
} PXA2xxRTCState;
|
|
|
|
|
|
|
|
static inline void pxa2xx_rtc_int_update(PXA2xxRTCState *s)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2011-02-25 14:13:38 +03:00
|
|
|
qemu_set_irq(s->rtc_irq, !!(s->rtsr & 0x2553));
|
2007-04-30 05:26:42 +04:00
|
|
|
}
|
|
|
|
|
2011-03-03 17:13:42 +03:00
|
|
|
static void pxa2xx_rtc_hzupdate(PXA2xxRTCState *s)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2012-03-30 14:31:22 +04:00
|
|
|
int64_t rt = qemu_get_clock_ms(rtc_clock);
|
2007-04-30 05:26:42 +04:00
|
|
|
s->last_rcnr += ((rt - s->last_hz) << 15) /
|
|
|
|
(1000 * ((s->rttr & 0xffff) + 1));
|
|
|
|
s->last_rdcr += ((rt - s->last_hz) << 15) /
|
|
|
|
(1000 * ((s->rttr & 0xffff) + 1));
|
|
|
|
s->last_hz = rt;
|
|
|
|
}
|
|
|
|
|
2011-03-03 17:13:42 +03:00
|
|
|
static void pxa2xx_rtc_swupdate(PXA2xxRTCState *s)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2012-03-30 14:31:22 +04:00
|
|
|
int64_t rt = qemu_get_clock_ms(rtc_clock);
|
2007-04-30 05:26:42 +04:00
|
|
|
if (s->rtsr & (1 << 12))
|
|
|
|
s->last_swcr += (rt - s->last_sw) / 10;
|
|
|
|
s->last_sw = rt;
|
|
|
|
}
|
|
|
|
|
2011-03-03 17:13:42 +03:00
|
|
|
static void pxa2xx_rtc_piupdate(PXA2xxRTCState *s)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2012-03-30 14:31:22 +04:00
|
|
|
int64_t rt = qemu_get_clock_ms(rtc_clock);
|
2007-04-30 05:26:42 +04:00
|
|
|
if (s->rtsr & (1 << 15))
|
|
|
|
s->last_swcr += rt - s->last_pi;
|
|
|
|
s->last_pi = rt;
|
|
|
|
}
|
|
|
|
|
2011-03-03 17:13:42 +03:00
|
|
|
static inline void pxa2xx_rtc_alarm_update(PXA2xxRTCState *s,
|
2007-04-30 05:26:42 +04:00
|
|
|
uint32_t rtsr)
|
|
|
|
{
|
|
|
|
if ((rtsr & (1 << 2)) && !(rtsr & (1 << 0)))
|
|
|
|
qemu_mod_timer(s->rtc_hz, s->last_hz +
|
|
|
|
(((s->rtar - s->last_rcnr) * 1000 *
|
|
|
|
((s->rttr & 0xffff) + 1)) >> 15));
|
|
|
|
else
|
|
|
|
qemu_del_timer(s->rtc_hz);
|
|
|
|
|
|
|
|
if ((rtsr & (1 << 5)) && !(rtsr & (1 << 4)))
|
|
|
|
qemu_mod_timer(s->rtc_rdal1, s->last_hz +
|
|
|
|
(((s->rdar1 - s->last_rdcr) * 1000 *
|
|
|
|
((s->rttr & 0xffff) + 1)) >> 15)); /* TODO: fixup */
|
|
|
|
else
|
|
|
|
qemu_del_timer(s->rtc_rdal1);
|
|
|
|
|
|
|
|
if ((rtsr & (1 << 7)) && !(rtsr & (1 << 6)))
|
|
|
|
qemu_mod_timer(s->rtc_rdal2, s->last_hz +
|
|
|
|
(((s->rdar2 - s->last_rdcr) * 1000 *
|
|
|
|
((s->rttr & 0xffff) + 1)) >> 15)); /* TODO: fixup */
|
|
|
|
else
|
|
|
|
qemu_del_timer(s->rtc_rdal2);
|
|
|
|
|
|
|
|
if ((rtsr & 0x1200) == 0x1200 && !(rtsr & (1 << 8)))
|
|
|
|
qemu_mod_timer(s->rtc_swal1, s->last_sw +
|
|
|
|
(s->swar1 - s->last_swcr) * 10); /* TODO: fixup */
|
|
|
|
else
|
|
|
|
qemu_del_timer(s->rtc_swal1);
|
|
|
|
|
|
|
|
if ((rtsr & 0x1800) == 0x1800 && !(rtsr & (1 << 10)))
|
|
|
|
qemu_mod_timer(s->rtc_swal2, s->last_sw +
|
|
|
|
(s->swar2 - s->last_swcr) * 10); /* TODO: fixup */
|
|
|
|
else
|
|
|
|
qemu_del_timer(s->rtc_swal2);
|
|
|
|
|
|
|
|
if ((rtsr & 0xc000) == 0xc000 && !(rtsr & (1 << 13)))
|
|
|
|
qemu_mod_timer(s->rtc_pi, s->last_pi +
|
|
|
|
(s->piar & 0xffff) - s->last_rtcpicr);
|
|
|
|
else
|
|
|
|
qemu_del_timer(s->rtc_pi);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void pxa2xx_rtc_hz_tick(void *opaque)
|
|
|
|
{
|
2011-03-03 17:13:42 +03:00
|
|
|
PXA2xxRTCState *s = (PXA2xxRTCState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
s->rtsr |= (1 << 0);
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
pxa2xx_rtc_int_update(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void pxa2xx_rtc_rdal1_tick(void *opaque)
|
|
|
|
{
|
2011-03-03 17:13:42 +03:00
|
|
|
PXA2xxRTCState *s = (PXA2xxRTCState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
s->rtsr |= (1 << 4);
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
pxa2xx_rtc_int_update(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void pxa2xx_rtc_rdal2_tick(void *opaque)
|
|
|
|
{
|
2011-03-03 17:13:42 +03:00
|
|
|
PXA2xxRTCState *s = (PXA2xxRTCState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
s->rtsr |= (1 << 6);
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
pxa2xx_rtc_int_update(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void pxa2xx_rtc_swal1_tick(void *opaque)
|
|
|
|
{
|
2011-03-03 17:13:42 +03:00
|
|
|
PXA2xxRTCState *s = (PXA2xxRTCState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
s->rtsr |= (1 << 8);
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
pxa2xx_rtc_int_update(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void pxa2xx_rtc_swal2_tick(void *opaque)
|
|
|
|
{
|
2011-03-03 17:13:42 +03:00
|
|
|
PXA2xxRTCState *s = (PXA2xxRTCState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
s->rtsr |= (1 << 10);
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
pxa2xx_rtc_int_update(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void pxa2xx_rtc_pi_tick(void *opaque)
|
|
|
|
{
|
2011-03-03 17:13:42 +03:00
|
|
|
PXA2xxRTCState *s = (PXA2xxRTCState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
s->rtsr |= (1 << 13);
|
|
|
|
pxa2xx_rtc_piupdate(s);
|
|
|
|
s->last_rtcpicr = 0;
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
pxa2xx_rtc_int_update(s);
|
|
|
|
}
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static uint64_t pxa2xx_rtc_read(void *opaque, target_phys_addr_t addr,
|
|
|
|
unsigned size)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2011-03-03 17:13:42 +03:00
|
|
|
PXA2xxRTCState *s = (PXA2xxRTCState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
switch (addr) {
|
|
|
|
case RTTR:
|
|
|
|
return s->rttr;
|
|
|
|
case RTSR:
|
|
|
|
return s->rtsr;
|
|
|
|
case RTAR:
|
|
|
|
return s->rtar;
|
|
|
|
case RDAR1:
|
|
|
|
return s->rdar1;
|
|
|
|
case RDAR2:
|
|
|
|
return s->rdar2;
|
|
|
|
case RYAR1:
|
|
|
|
return s->ryar1;
|
|
|
|
case RYAR2:
|
|
|
|
return s->ryar2;
|
|
|
|
case SWAR1:
|
|
|
|
return s->swar1;
|
|
|
|
case SWAR2:
|
|
|
|
return s->swar2;
|
|
|
|
case PIAR:
|
|
|
|
return s->piar;
|
|
|
|
case RCNR:
|
2012-03-30 14:31:22 +04:00
|
|
|
return s->last_rcnr + ((qemu_get_clock_ms(rtc_clock) - s->last_hz) << 15) /
|
2007-04-30 05:26:42 +04:00
|
|
|
(1000 * ((s->rttr & 0xffff) + 1));
|
|
|
|
case RDCR:
|
2012-03-30 14:31:22 +04:00
|
|
|
return s->last_rdcr + ((qemu_get_clock_ms(rtc_clock) - s->last_hz) << 15) /
|
2007-04-30 05:26:42 +04:00
|
|
|
(1000 * ((s->rttr & 0xffff) + 1));
|
|
|
|
case RYCR:
|
|
|
|
return s->last_rycr;
|
|
|
|
case SWCR:
|
|
|
|
if (s->rtsr & (1 << 12))
|
2012-03-30 14:31:22 +04:00
|
|
|
return s->last_swcr + (qemu_get_clock_ms(rtc_clock) - s->last_sw) / 10;
|
2007-04-30 05:26:42 +04:00
|
|
|
else
|
|
|
|
return s->last_swcr;
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-10-02 01:12:16 +04:00
|
|
|
static void pxa2xx_rtc_write(void *opaque, target_phys_addr_t addr,
|
2011-09-25 19:19:19 +04:00
|
|
|
uint64_t value64, unsigned size)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2011-03-03 17:13:42 +03:00
|
|
|
PXA2xxRTCState *s = (PXA2xxRTCState *) opaque;
|
2011-09-25 19:19:19 +04:00
|
|
|
uint32_t value = value64;
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
switch (addr) {
|
|
|
|
case RTTR:
|
|
|
|
if (!(s->rttr & (1 << 31))) {
|
|
|
|
pxa2xx_rtc_hzupdate(s);
|
|
|
|
s->rttr = value;
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RTSR:
|
|
|
|
if ((s->rtsr ^ value) & (1 << 15))
|
|
|
|
pxa2xx_rtc_piupdate(s);
|
|
|
|
|
|
|
|
if ((s->rtsr ^ value) & (1 << 12))
|
|
|
|
pxa2xx_rtc_swupdate(s);
|
|
|
|
|
|
|
|
if (((s->rtsr ^ value) & 0x4aac) | (value & ~0xdaac))
|
|
|
|
pxa2xx_rtc_alarm_update(s, value);
|
|
|
|
|
|
|
|
s->rtsr = (value & 0xdaac) | (s->rtsr & ~(value & ~0xdaac));
|
|
|
|
pxa2xx_rtc_int_update(s);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RTAR:
|
|
|
|
s->rtar = value;
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RDAR1:
|
|
|
|
s->rdar1 = value;
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RDAR2:
|
|
|
|
s->rdar2 = value;
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RYAR1:
|
|
|
|
s->ryar1 = value;
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RYAR2:
|
|
|
|
s->ryar2 = value;
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SWAR1:
|
|
|
|
pxa2xx_rtc_swupdate(s);
|
|
|
|
s->swar1 = value;
|
|
|
|
s->last_swcr = 0;
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SWAR2:
|
|
|
|
s->swar2 = value;
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PIAR:
|
|
|
|
s->piar = value;
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RCNR:
|
|
|
|
pxa2xx_rtc_hzupdate(s);
|
|
|
|
s->last_rcnr = value;
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RDCR:
|
|
|
|
pxa2xx_rtc_hzupdate(s);
|
|
|
|
s->last_rdcr = value;
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RYCR:
|
|
|
|
s->last_rycr = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SWCR:
|
|
|
|
pxa2xx_rtc_swupdate(s);
|
|
|
|
s->last_swcr = value;
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RTCPICR:
|
|
|
|
pxa2xx_rtc_piupdate(s);
|
|
|
|
s->last_rtcpicr = value & 0xffff;
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static const MemoryRegionOps pxa2xx_rtc_ops = {
|
|
|
|
.read = pxa2xx_rtc_read,
|
|
|
|
.write = pxa2xx_rtc_write,
|
|
|
|
.endianness = DEVICE_NATIVE_ENDIAN,
|
2007-05-24 22:50:09 +04:00
|
|
|
};
|
|
|
|
|
2011-03-03 17:13:42 +03:00
|
|
|
static int pxa2xx_rtc_init(SysBusDevice *dev)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2011-03-03 17:13:42 +03:00
|
|
|
PXA2xxRTCState *s = FROM_SYSBUS(PXA2xxRTCState, dev);
|
2008-02-17 14:42:19 +03:00
|
|
|
struct tm tm;
|
2007-04-30 05:26:42 +04:00
|
|
|
int wom;
|
|
|
|
|
|
|
|
s->rttr = 0x7fff;
|
|
|
|
s->rtsr = 0;
|
|
|
|
|
2008-02-17 14:42:19 +03:00
|
|
|
qemu_get_timedate(&tm, 0);
|
|
|
|
wom = ((tm.tm_mday - 1) / 7) + 1;
|
|
|
|
|
2008-12-05 00:34:52 +03:00
|
|
|
s->last_rcnr = (uint32_t) mktimegm(&tm);
|
2008-02-17 14:42:19 +03:00
|
|
|
s->last_rdcr = (wom << 20) | ((tm.tm_wday + 1) << 17) |
|
|
|
|
(tm.tm_hour << 12) | (tm.tm_min << 6) | tm.tm_sec;
|
|
|
|
s->last_rycr = ((tm.tm_year + 1900) << 9) |
|
|
|
|
((tm.tm_mon + 1) << 5) | tm.tm_mday;
|
|
|
|
s->last_swcr = (tm.tm_hour << 19) |
|
|
|
|
(tm.tm_min << 13) | (tm.tm_sec << 7);
|
2007-04-30 05:26:42 +04:00
|
|
|
s->last_rtcpicr = 0;
|
2012-03-30 14:31:22 +04:00
|
|
|
s->last_hz = s->last_sw = s->last_pi = qemu_get_clock_ms(rtc_clock);
|
|
|
|
|
|
|
|
s->rtc_hz = qemu_new_timer_ms(rtc_clock, pxa2xx_rtc_hz_tick, s);
|
|
|
|
s->rtc_rdal1 = qemu_new_timer_ms(rtc_clock, pxa2xx_rtc_rdal1_tick, s);
|
|
|
|
s->rtc_rdal2 = qemu_new_timer_ms(rtc_clock, pxa2xx_rtc_rdal2_tick, s);
|
|
|
|
s->rtc_swal1 = qemu_new_timer_ms(rtc_clock, pxa2xx_rtc_swal1_tick, s);
|
|
|
|
s->rtc_swal2 = qemu_new_timer_ms(rtc_clock, pxa2xx_rtc_swal2_tick, s);
|
|
|
|
s->rtc_pi = qemu_new_timer_ms(rtc_clock, pxa2xx_rtc_pi_tick, s);
|
2011-02-25 14:13:38 +03:00
|
|
|
|
2011-03-03 17:13:42 +03:00
|
|
|
sysbus_init_irq(dev, &s->rtc_irq);
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
memory_region_init_io(&s->iomem, &pxa2xx_rtc_ops, s, "pxa2xx-rtc", 0x10000);
|
2011-11-27 13:38:10 +04:00
|
|
|
sysbus_init_mmio(dev, &s->iomem);
|
2011-03-03 17:13:42 +03:00
|
|
|
|
|
|
|
return 0;
|
2007-04-30 05:26:42 +04:00
|
|
|
}
|
|
|
|
|
2011-03-03 17:13:42 +03:00
|
|
|
static void pxa2xx_rtc_pre_save(void *opaque)
|
2007-05-24 22:50:09 +04:00
|
|
|
{
|
2011-03-03 17:13:42 +03:00
|
|
|
PXA2xxRTCState *s = (PXA2xxRTCState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2007-05-24 22:50:09 +04:00
|
|
|
pxa2xx_rtc_hzupdate(s);
|
|
|
|
pxa2xx_rtc_piupdate(s);
|
|
|
|
pxa2xx_rtc_swupdate(s);
|
2011-03-03 17:13:42 +03:00
|
|
|
}
|
2007-05-24 22:50:09 +04:00
|
|
|
|
2011-03-03 17:13:42 +03:00
|
|
|
static int pxa2xx_rtc_post_load(void *opaque, int version_id)
|
2007-05-24 22:50:09 +04:00
|
|
|
{
|
2011-03-03 17:13:42 +03:00
|
|
|
PXA2xxRTCState *s = (PXA2xxRTCState *) opaque;
|
2007-05-24 22:50:09 +04:00
|
|
|
|
|
|
|
pxa2xx_rtc_alarm_update(s, s->rtsr);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2011-03-03 17:13:42 +03:00
|
|
|
static const VMStateDescription vmstate_pxa2xx_rtc_regs = {
|
|
|
|
.name = "pxa2xx_rtc",
|
|
|
|
.version_id = 0,
|
|
|
|
.minimum_version_id = 0,
|
|
|
|
.minimum_version_id_old = 0,
|
|
|
|
.pre_save = pxa2xx_rtc_pre_save,
|
|
|
|
.post_load = pxa2xx_rtc_post_load,
|
|
|
|
.fields = (VMStateField[]) {
|
|
|
|
VMSTATE_UINT32(rttr, PXA2xxRTCState),
|
|
|
|
VMSTATE_UINT32(rtsr, PXA2xxRTCState),
|
|
|
|
VMSTATE_UINT32(rtar, PXA2xxRTCState),
|
|
|
|
VMSTATE_UINT32(rdar1, PXA2xxRTCState),
|
|
|
|
VMSTATE_UINT32(rdar2, PXA2xxRTCState),
|
|
|
|
VMSTATE_UINT32(ryar1, PXA2xxRTCState),
|
|
|
|
VMSTATE_UINT32(ryar2, PXA2xxRTCState),
|
|
|
|
VMSTATE_UINT32(swar1, PXA2xxRTCState),
|
|
|
|
VMSTATE_UINT32(swar2, PXA2xxRTCState),
|
|
|
|
VMSTATE_UINT32(piar, PXA2xxRTCState),
|
|
|
|
VMSTATE_UINT32(last_rcnr, PXA2xxRTCState),
|
|
|
|
VMSTATE_UINT32(last_rdcr, PXA2xxRTCState),
|
|
|
|
VMSTATE_UINT32(last_rycr, PXA2xxRTCState),
|
|
|
|
VMSTATE_UINT32(last_swcr, PXA2xxRTCState),
|
|
|
|
VMSTATE_UINT32(last_rtcpicr, PXA2xxRTCState),
|
|
|
|
VMSTATE_INT64(last_hz, PXA2xxRTCState),
|
|
|
|
VMSTATE_INT64(last_sw, PXA2xxRTCState),
|
|
|
|
VMSTATE_INT64(last_pi, PXA2xxRTCState),
|
|
|
|
VMSTATE_END_OF_LIST(),
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2012-01-24 23:12:29 +04:00
|
|
|
static void pxa2xx_rtc_sysbus_class_init(ObjectClass *klass, void *data)
|
|
|
|
{
|
2011-12-08 07:34:16 +04:00
|
|
|
DeviceClass *dc = DEVICE_CLASS(klass);
|
2012-01-24 23:12:29 +04:00
|
|
|
SysBusDeviceClass *k = SYS_BUS_DEVICE_CLASS(klass);
|
|
|
|
|
|
|
|
k->init = pxa2xx_rtc_init;
|
2011-12-08 07:34:16 +04:00
|
|
|
dc->desc = "PXA2xx RTC Controller";
|
|
|
|
dc->vmsd = &vmstate_pxa2xx_rtc_regs;
|
2012-01-24 23:12:29 +04:00
|
|
|
}
|
|
|
|
|
2011-12-08 07:34:16 +04:00
|
|
|
static TypeInfo pxa2xx_rtc_sysbus_info = {
|
|
|
|
.name = "pxa2xx_rtc",
|
|
|
|
.parent = TYPE_SYS_BUS_DEVICE,
|
|
|
|
.instance_size = sizeof(PXA2xxRTCState),
|
|
|
|
.class_init = pxa2xx_rtc_sysbus_class_init,
|
2011-03-03 17:13:42 +03:00
|
|
|
};
|
|
|
|
|
2007-05-24 01:47:51 +04:00
|
|
|
/* I2C Interface */
|
2009-05-15 01:35:08 +04:00
|
|
|
typedef struct {
|
2011-12-05 06:28:27 +04:00
|
|
|
I2CSlave i2c;
|
2009-05-15 01:35:08 +04:00
|
|
|
PXA2xxI2CState *host;
|
|
|
|
} PXA2xxI2CSlaveState;
|
|
|
|
|
2009-05-10 04:44:56 +04:00
|
|
|
struct PXA2xxI2CState {
|
2011-02-11 23:57:39 +03:00
|
|
|
SysBusDevice busdev;
|
2011-09-25 19:19:19 +04:00
|
|
|
MemoryRegion iomem;
|
2009-05-15 01:35:08 +04:00
|
|
|
PXA2xxI2CSlaveState *slave;
|
2007-05-24 01:47:51 +04:00
|
|
|
i2c_bus *bus;
|
|
|
|
qemu_irq irq;
|
2011-02-11 23:57:39 +03:00
|
|
|
uint32_t offset;
|
|
|
|
uint32_t region_size;
|
2007-05-24 01:47:51 +04:00
|
|
|
|
|
|
|
uint16_t control;
|
|
|
|
uint16_t status;
|
|
|
|
uint8_t ibmr;
|
|
|
|
uint8_t data;
|
|
|
|
};
|
|
|
|
|
|
|
|
#define IBMR 0x80 /* I2C Bus Monitor register */
|
|
|
|
#define IDBR 0x88 /* I2C Data Buffer register */
|
|
|
|
#define ICR 0x90 /* I2C Control register */
|
|
|
|
#define ISR 0x98 /* I2C Status register */
|
|
|
|
#define ISAR 0xa0 /* I2C Slave Address register */
|
|
|
|
|
2009-05-10 04:44:56 +04:00
|
|
|
static void pxa2xx_i2c_update(PXA2xxI2CState *s)
|
2007-05-24 01:47:51 +04:00
|
|
|
{
|
|
|
|
uint16_t level = 0;
|
|
|
|
level |= s->status & s->control & (1 << 10); /* BED */
|
|
|
|
level |= (s->status & (1 << 7)) && (s->control & (1 << 9)); /* IRF */
|
|
|
|
level |= (s->status & (1 << 6)) && (s->control & (1 << 8)); /* ITE */
|
|
|
|
level |= s->status & (1 << 9); /* SAD */
|
|
|
|
qemu_set_irq(s->irq, !!level);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* These are only stubs now. */
|
2011-12-05 06:28:27 +04:00
|
|
|
static void pxa2xx_i2c_event(I2CSlave *i2c, enum i2c_event event)
|
2007-05-24 01:47:51 +04:00
|
|
|
{
|
2009-05-15 01:35:08 +04:00
|
|
|
PXA2xxI2CSlaveState *slave = FROM_I2C_SLAVE(PXA2xxI2CSlaveState, i2c);
|
|
|
|
PXA2xxI2CState *s = slave->host;
|
2007-05-24 01:47:51 +04:00
|
|
|
|
|
|
|
switch (event) {
|
|
|
|
case I2C_START_SEND:
|
|
|
|
s->status |= (1 << 9); /* set SAD */
|
|
|
|
s->status &= ~(1 << 0); /* clear RWM */
|
|
|
|
break;
|
|
|
|
case I2C_START_RECV:
|
|
|
|
s->status |= (1 << 9); /* set SAD */
|
|
|
|
s->status |= 1 << 0; /* set RWM */
|
|
|
|
break;
|
|
|
|
case I2C_FINISH:
|
|
|
|
s->status |= (1 << 4); /* set SSD */
|
|
|
|
break;
|
|
|
|
case I2C_NACK:
|
|
|
|
s->status |= 1 << 1; /* set ACKNAK */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
pxa2xx_i2c_update(s);
|
|
|
|
}
|
|
|
|
|
2011-12-05 06:28:27 +04:00
|
|
|
static int pxa2xx_i2c_rx(I2CSlave *i2c)
|
2007-05-24 01:47:51 +04:00
|
|
|
{
|
2009-05-15 01:35:08 +04:00
|
|
|
PXA2xxI2CSlaveState *slave = FROM_I2C_SLAVE(PXA2xxI2CSlaveState, i2c);
|
|
|
|
PXA2xxI2CState *s = slave->host;
|
2007-05-24 01:47:51 +04:00
|
|
|
if ((s->control & (1 << 14)) || !(s->control & (1 << 6)))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (s->status & (1 << 0)) { /* RWM */
|
|
|
|
s->status |= 1 << 6; /* set ITE */
|
|
|
|
}
|
|
|
|
pxa2xx_i2c_update(s);
|
|
|
|
|
|
|
|
return s->data;
|
|
|
|
}
|
|
|
|
|
2011-12-05 06:28:27 +04:00
|
|
|
static int pxa2xx_i2c_tx(I2CSlave *i2c, uint8_t data)
|
2007-05-24 01:47:51 +04:00
|
|
|
{
|
2009-05-15 01:35:08 +04:00
|
|
|
PXA2xxI2CSlaveState *slave = FROM_I2C_SLAVE(PXA2xxI2CSlaveState, i2c);
|
|
|
|
PXA2xxI2CState *s = slave->host;
|
2007-05-24 01:47:51 +04:00
|
|
|
if ((s->control & (1 << 14)) || !(s->control & (1 << 6)))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (!(s->status & (1 << 0))) { /* RWM */
|
|
|
|
s->status |= 1 << 7; /* set IRF */
|
|
|
|
s->data = data;
|
|
|
|
}
|
|
|
|
pxa2xx_i2c_update(s);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static uint64_t pxa2xx_i2c_read(void *opaque, target_phys_addr_t addr,
|
|
|
|
unsigned size)
|
2007-05-24 01:47:51 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxI2CState *s = (PXA2xxI2CState *) opaque;
|
2007-05-24 01:47:51 +04:00
|
|
|
|
2008-12-07 04:49:28 +03:00
|
|
|
addr -= s->offset;
|
2007-05-24 01:47:51 +04:00
|
|
|
switch (addr) {
|
|
|
|
case ICR:
|
|
|
|
return s->control;
|
|
|
|
case ISR:
|
|
|
|
return s->status | (i2c_bus_busy(s->bus) << 2);
|
|
|
|
case ISAR:
|
2009-05-15 01:35:08 +04:00
|
|
|
return s->slave->i2c.address;
|
2007-05-24 01:47:51 +04:00
|
|
|
case IDBR:
|
|
|
|
return s->data;
|
|
|
|
case IBMR:
|
|
|
|
if (s->status & (1 << 2))
|
|
|
|
s->ibmr ^= 3; /* Fake SCL and SDA pin changes */
|
|
|
|
else
|
|
|
|
s->ibmr = 0;
|
|
|
|
return s->ibmr;
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-10-02 01:12:16 +04:00
|
|
|
static void pxa2xx_i2c_write(void *opaque, target_phys_addr_t addr,
|
2011-09-25 19:19:19 +04:00
|
|
|
uint64_t value64, unsigned size)
|
2007-05-24 01:47:51 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxI2CState *s = (PXA2xxI2CState *) opaque;
|
2011-09-25 19:19:19 +04:00
|
|
|
uint32_t value = value64;
|
2007-05-24 01:47:51 +04:00
|
|
|
int ack;
|
|
|
|
|
2008-12-07 04:49:28 +03:00
|
|
|
addr -= s->offset;
|
2007-05-24 01:47:51 +04:00
|
|
|
switch (addr) {
|
|
|
|
case ICR:
|
|
|
|
s->control = value & 0xfff7;
|
|
|
|
if ((value & (1 << 3)) && (value & (1 << 6))) { /* TB and IUE */
|
|
|
|
/* TODO: slave mode */
|
|
|
|
if (value & (1 << 0)) { /* START condition */
|
|
|
|
if (s->data & 1)
|
|
|
|
s->status |= 1 << 0; /* set RWM */
|
|
|
|
else
|
|
|
|
s->status &= ~(1 << 0); /* clear RWM */
|
|
|
|
ack = !i2c_start_transfer(s->bus, s->data >> 1, s->data & 1);
|
|
|
|
} else {
|
|
|
|
if (s->status & (1 << 0)) { /* RWM */
|
|
|
|
s->data = i2c_recv(s->bus);
|
|
|
|
if (value & (1 << 2)) /* ACKNAK */
|
|
|
|
i2c_nack(s->bus);
|
|
|
|
ack = 1;
|
|
|
|
} else
|
|
|
|
ack = !i2c_send(s->bus, s->data);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (value & (1 << 1)) /* STOP condition */
|
|
|
|
i2c_end_transfer(s->bus);
|
|
|
|
|
|
|
|
if (ack) {
|
|
|
|
if (value & (1 << 0)) /* START condition */
|
|
|
|
s->status |= 1 << 6; /* set ITE */
|
|
|
|
else
|
|
|
|
if (s->status & (1 << 0)) /* RWM */
|
|
|
|
s->status |= 1 << 7; /* set IRF */
|
|
|
|
else
|
|
|
|
s->status |= 1 << 6; /* set ITE */
|
|
|
|
s->status &= ~(1 << 1); /* clear ACKNAK */
|
|
|
|
} else {
|
|
|
|
s->status |= 1 << 6; /* set ITE */
|
|
|
|
s->status |= 1 << 10; /* set BED */
|
|
|
|
s->status |= 1 << 1; /* set ACKNAK */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!(value & (1 << 3)) && (value & (1 << 6))) /* !TB and IUE */
|
|
|
|
if (value & (1 << 4)) /* MA */
|
|
|
|
i2c_end_transfer(s->bus);
|
|
|
|
pxa2xx_i2c_update(s);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ISR:
|
|
|
|
s->status &= ~(value & 0x07f0);
|
|
|
|
pxa2xx_i2c_update(s);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ISAR:
|
2009-05-15 01:35:08 +04:00
|
|
|
i2c_set_slave_address(&s->slave->i2c, value & 0x7f);
|
2007-05-24 01:47:51 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case IDBR:
|
|
|
|
s->data = value & 0xff;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static const MemoryRegionOps pxa2xx_i2c_ops = {
|
|
|
|
.read = pxa2xx_i2c_read,
|
|
|
|
.write = pxa2xx_i2c_write,
|
|
|
|
.endianness = DEVICE_NATIVE_ENDIAN,
|
2007-05-24 01:47:51 +04:00
|
|
|
};
|
|
|
|
|
2009-09-30 00:48:34 +04:00
|
|
|
static const VMStateDescription vmstate_pxa2xx_i2c_slave = {
|
|
|
|
.name = "pxa2xx_i2c_slave",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.minimum_version_id_old = 1,
|
|
|
|
.fields = (VMStateField []) {
|
|
|
|
VMSTATE_I2C_SLAVE(i2c, PXA2xxI2CSlaveState),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
2007-05-24 22:50:09 +04:00
|
|
|
|
2009-09-30 00:48:34 +04:00
|
|
|
static const VMStateDescription vmstate_pxa2xx_i2c = {
|
|
|
|
.name = "pxa2xx_i2c",
|
|
|
|
.version_id = 1,
|
|
|
|
.minimum_version_id = 1,
|
|
|
|
.minimum_version_id_old = 1,
|
|
|
|
.fields = (VMStateField []) {
|
|
|
|
VMSTATE_UINT16(control, PXA2xxI2CState),
|
|
|
|
VMSTATE_UINT16(status, PXA2xxI2CState),
|
|
|
|
VMSTATE_UINT8(ibmr, PXA2xxI2CState),
|
|
|
|
VMSTATE_UINT8(data, PXA2xxI2CState),
|
|
|
|
VMSTATE_STRUCT_POINTER(slave, PXA2xxI2CState,
|
2011-01-13 18:37:12 +03:00
|
|
|
vmstate_pxa2xx_i2c_slave, PXA2xxI2CSlaveState *),
|
2009-09-30 00:48:34 +04:00
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
2007-05-24 22:50:09 +04:00
|
|
|
|
2011-12-05 06:28:27 +04:00
|
|
|
static int pxa2xx_i2c_slave_init(I2CSlave *i2c)
|
2009-05-15 01:35:08 +04:00
|
|
|
{
|
|
|
|
/* Nothing to do. */
|
2009-08-14 12:36:05 +04:00
|
|
|
return 0;
|
2009-05-15 01:35:08 +04:00
|
|
|
}
|
|
|
|
|
2012-01-24 23:12:29 +04:00
|
|
|
static void pxa2xx_i2c_slave_class_init(ObjectClass *klass, void *data)
|
2011-12-05 06:39:20 +04:00
|
|
|
{
|
|
|
|
I2CSlaveClass *k = I2C_SLAVE_CLASS(klass);
|
|
|
|
|
|
|
|
k->init = pxa2xx_i2c_slave_init;
|
|
|
|
k->event = pxa2xx_i2c_event;
|
|
|
|
k->recv = pxa2xx_i2c_rx;
|
|
|
|
k->send = pxa2xx_i2c_tx;
|
|
|
|
}
|
|
|
|
|
2011-12-08 07:34:16 +04:00
|
|
|
static TypeInfo pxa2xx_i2c_slave_info = {
|
|
|
|
.name = "pxa2xx-i2c-slave",
|
|
|
|
.parent = TYPE_I2C_SLAVE,
|
|
|
|
.instance_size = sizeof(PXA2xxI2CSlaveState),
|
|
|
|
.class_init = pxa2xx_i2c_slave_class_init,
|
2009-05-15 01:35:08 +04:00
|
|
|
};
|
|
|
|
|
2009-10-02 01:12:16 +04:00
|
|
|
PXA2xxI2CState *pxa2xx_i2c_init(target_phys_addr_t base,
|
2008-12-07 04:49:28 +03:00
|
|
|
qemu_irq irq, uint32_t region_size)
|
2007-05-24 01:47:51 +04:00
|
|
|
{
|
2009-05-15 01:35:08 +04:00
|
|
|
DeviceState *dev;
|
2011-02-11 23:57:39 +03:00
|
|
|
SysBusDevice *i2c_dev;
|
|
|
|
PXA2xxI2CState *s;
|
|
|
|
|
|
|
|
i2c_dev = sysbus_from_qdev(qdev_create(NULL, "pxa2xx_i2c"));
|
|
|
|
qdev_prop_set_uint32(&i2c_dev->qdev, "size", region_size + 1);
|
hw/pxa2xx.c: Fix handling of pxa2xx_i2c variable offset within region
The pxa2xx I2C controller can have its registers at an arbitrary offset
within the MemoryRegion it creates. We use this to create two controllers,
one which covers a region of size 0x10000 with registers starting at an
offset 0x1600 into that region, and a second one which covers a region
of size just 0x100 with the registers starting at the base of the region.
The implementation of this offsetting uses two qdev properties, "offset"
(which sets the offset which must be subtracted from the address to
get the offset into the actual register bank) and "size", which is the
size of the MemoryRegion. We were actually using "offset" for two
purposes: firstly the required one of handling the registers not being
at the base of the MemoryRegion, and secondly as a workaround for a
deficiency of QEMU. Until commit 5312bd8b3, if a MemoryRegion was mapped
at a non-page boundary, the address passed into the read and write
functions would be the offset from the start of the page, not the
offset from the start of the MemoryRegion. So when calculating the value
to set the "offset" qdev property we included a rounding to a page
boundary.
Following commit 5312bd8b3 MemoryRegion read/write functions are now
correctly passed the offset from the base of the region, and our
workaround now means we're subtracting too much from addresses, resulting
in warnings like "pxa2xx_i2c_read: Bad register 0xffffff90".
The fix for this is simply to remove the rounding to a page boundary;
this allows us to slightly simplify the expression since
base - (base & (~region_size)) == base & region_size
The qdev property "offset" itself must remain because it is still
performing its primary job of handling register banks not being at
the base of the MemoryRegion.
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
Reviewed-by: Andreas Färber <afaerber@suse.de>
2012-03-14 19:37:53 +04:00
|
|
|
qdev_prop_set_uint32(&i2c_dev->qdev, "offset", base & region_size);
|
2011-02-11 23:57:39 +03:00
|
|
|
|
|
|
|
qdev_init_nofail(&i2c_dev->qdev);
|
|
|
|
|
|
|
|
sysbus_mmio_map(i2c_dev, 0, base & ~region_size);
|
|
|
|
sysbus_connect_irq(i2c_dev, 0, irq);
|
2009-05-15 01:35:08 +04:00
|
|
|
|
2011-02-11 23:57:39 +03:00
|
|
|
s = FROM_SYSBUS(PXA2xxI2CState, i2c_dev);
|
2008-07-02 03:16:53 +04:00
|
|
|
/* FIXME: Should the slave device really be on a separate bus? */
|
2009-05-23 03:05:19 +04:00
|
|
|
dev = i2c_create_slave(i2c_init_bus(NULL, "dummy"), "pxa2xx-i2c-slave", 0);
|
2009-05-15 01:35:08 +04:00
|
|
|
s->slave = FROM_I2C_SLAVE(PXA2xxI2CSlaveState, I2C_SLAVE_FROM_QDEV(dev));
|
|
|
|
s->slave->host = s;
|
2007-05-24 01:47:51 +04:00
|
|
|
|
2011-02-11 23:57:39 +03:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int pxa2xx_i2c_initfn(SysBusDevice *dev)
|
|
|
|
{
|
|
|
|
PXA2xxI2CState *s = FROM_SYSBUS(PXA2xxI2CState, dev);
|
|
|
|
|
|
|
|
s->bus = i2c_init_bus(&dev->qdev, "i2c");
|
2007-05-24 01:47:51 +04:00
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
memory_region_init_io(&s->iomem, &pxa2xx_i2c_ops, s,
|
|
|
|
"pxa2xx-i2x", s->region_size);
|
2011-11-27 13:38:10 +04:00
|
|
|
sysbus_init_mmio(dev, &s->iomem);
|
2011-02-11 23:57:39 +03:00
|
|
|
sysbus_init_irq(dev, &s->irq);
|
2007-05-24 22:50:09 +04:00
|
|
|
|
2011-02-11 23:57:39 +03:00
|
|
|
return 0;
|
2007-05-24 01:47:51 +04:00
|
|
|
}
|
|
|
|
|
2009-05-10 04:44:56 +04:00
|
|
|
i2c_bus *pxa2xx_i2c_bus(PXA2xxI2CState *s)
|
2007-05-24 01:47:51 +04:00
|
|
|
{
|
|
|
|
return s->bus;
|
|
|
|
}
|
|
|
|
|
2012-01-24 23:12:29 +04:00
|
|
|
static Property pxa2xx_i2c_properties[] = {
|
|
|
|
DEFINE_PROP_UINT32("size", PXA2xxI2CState, region_size, 0x10000),
|
|
|
|
DEFINE_PROP_UINT32("offset", PXA2xxI2CState, offset, 0),
|
|
|
|
DEFINE_PROP_END_OF_LIST(),
|
|
|
|
};
|
|
|
|
|
|
|
|
static void pxa2xx_i2c_class_init(ObjectClass *klass, void *data)
|
|
|
|
{
|
2011-12-08 07:34:16 +04:00
|
|
|
DeviceClass *dc = DEVICE_CLASS(klass);
|
2012-01-24 23:12:29 +04:00
|
|
|
SysBusDeviceClass *k = SYS_BUS_DEVICE_CLASS(klass);
|
|
|
|
|
|
|
|
k->init = pxa2xx_i2c_initfn;
|
2011-12-08 07:34:16 +04:00
|
|
|
dc->desc = "PXA2xx I2C Bus Controller";
|
|
|
|
dc->vmsd = &vmstate_pxa2xx_i2c;
|
|
|
|
dc->props = pxa2xx_i2c_properties;
|
2012-01-24 23:12:29 +04:00
|
|
|
}
|
|
|
|
|
2011-12-08 07:34:16 +04:00
|
|
|
static TypeInfo pxa2xx_i2c_info = {
|
|
|
|
.name = "pxa2xx_i2c",
|
|
|
|
.parent = TYPE_SYS_BUS_DEVICE,
|
|
|
|
.instance_size = sizeof(PXA2xxI2CState),
|
|
|
|
.class_init = pxa2xx_i2c_class_init,
|
2011-02-11 23:57:39 +03:00
|
|
|
};
|
|
|
|
|
2007-04-30 05:26:42 +04:00
|
|
|
/* PXA Inter-IC Sound Controller */
|
2009-05-10 04:44:56 +04:00
|
|
|
static void pxa2xx_i2s_reset(PXA2xxI2SState *i2s)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
|
|
|
i2s->rx_len = 0;
|
|
|
|
i2s->tx_len = 0;
|
|
|
|
i2s->fifo_len = 0;
|
|
|
|
i2s->clk = 0x1a;
|
|
|
|
i2s->control[0] = 0x00;
|
|
|
|
i2s->control[1] = 0x00;
|
|
|
|
i2s->status = 0x00;
|
|
|
|
i2s->mask = 0x00;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define SACR_TFTH(val) ((val >> 8) & 0xf)
|
|
|
|
#define SACR_RFTH(val) ((val >> 12) & 0xf)
|
|
|
|
#define SACR_DREC(val) (val & (1 << 3))
|
|
|
|
#define SACR_DPRL(val) (val & (1 << 4))
|
|
|
|
|
2009-05-10 04:44:56 +04:00
|
|
|
static inline void pxa2xx_i2s_update(PXA2xxI2SState *i2s)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
|
|
|
int rfs, tfs;
|
|
|
|
rfs = SACR_RFTH(i2s->control[0]) < i2s->rx_len &&
|
|
|
|
!SACR_DREC(i2s->control[1]);
|
|
|
|
tfs = (i2s->tx_len || i2s->fifo_len < SACR_TFTH(i2s->control[0])) &&
|
|
|
|
i2s->enable && !SACR_DPRL(i2s->control[1]);
|
|
|
|
|
2011-03-03 17:04:51 +03:00
|
|
|
qemu_set_irq(i2s->rx_dma, rfs);
|
|
|
|
qemu_set_irq(i2s->tx_dma, tfs);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
i2s->status &= 0xe0;
|
2007-07-28 01:49:15 +04:00
|
|
|
if (i2s->fifo_len < 16 || !i2s->enable)
|
|
|
|
i2s->status |= 1 << 0; /* TNF */
|
2007-04-30 05:26:42 +04:00
|
|
|
if (i2s->rx_len)
|
|
|
|
i2s->status |= 1 << 1; /* RNE */
|
|
|
|
if (i2s->enable)
|
|
|
|
i2s->status |= 1 << 2; /* BSY */
|
|
|
|
if (tfs)
|
|
|
|
i2s->status |= 1 << 3; /* TFS */
|
|
|
|
if (rfs)
|
|
|
|
i2s->status |= 1 << 4; /* RFS */
|
|
|
|
if (!(i2s->tx_len && i2s->enable))
|
|
|
|
i2s->status |= i2s->fifo_len << 8; /* TFL */
|
|
|
|
i2s->status |= MAX(i2s->rx_len, 0xf) << 12; /* RFL */
|
|
|
|
|
|
|
|
qemu_set_irq(i2s->irq, i2s->status & i2s->mask);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define SACR0 0x00 /* Serial Audio Global Control register */
|
|
|
|
#define SACR1 0x04 /* Serial Audio I2S/MSB-Justified Control register */
|
|
|
|
#define SASR0 0x0c /* Serial Audio Interface and FIFO Status register */
|
|
|
|
#define SAIMR 0x14 /* Serial Audio Interrupt Mask register */
|
|
|
|
#define SAICR 0x18 /* Serial Audio Interrupt Clear register */
|
|
|
|
#define SADIV 0x60 /* Serial Audio Clock Divider register */
|
|
|
|
#define SADR 0x80 /* Serial Audio Data register */
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static uint64_t pxa2xx_i2s_read(void *opaque, target_phys_addr_t addr,
|
|
|
|
unsigned size)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxI2SState *s = (PXA2xxI2SState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
switch (addr) {
|
|
|
|
case SACR0:
|
|
|
|
return s->control[0];
|
|
|
|
case SACR1:
|
|
|
|
return s->control[1];
|
|
|
|
case SASR0:
|
|
|
|
return s->status;
|
|
|
|
case SAIMR:
|
|
|
|
return s->mask;
|
|
|
|
case SAICR:
|
|
|
|
return 0;
|
|
|
|
case SADIV:
|
|
|
|
return s->clk;
|
|
|
|
case SADR:
|
|
|
|
if (s->rx_len > 0) {
|
|
|
|
s->rx_len --;
|
|
|
|
pxa2xx_i2s_update(s);
|
|
|
|
return s->codec_in(s->opaque);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-10-02 01:12:16 +04:00
|
|
|
static void pxa2xx_i2s_write(void *opaque, target_phys_addr_t addr,
|
2011-09-25 19:19:19 +04:00
|
|
|
uint64_t value, unsigned size)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxI2SState *s = (PXA2xxI2SState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
uint32_t *sample;
|
|
|
|
|
|
|
|
switch (addr) {
|
|
|
|
case SACR0:
|
|
|
|
if (value & (1 << 3)) /* RST */
|
|
|
|
pxa2xx_i2s_reset(s);
|
|
|
|
s->control[0] = value & 0xff3d;
|
|
|
|
if (!s->enable && (value & 1) && s->tx_len) { /* ENB */
|
|
|
|
for (sample = s->fifo; s->fifo_len > 0; s->fifo_len --, sample ++)
|
|
|
|
s->codec_out(s->opaque, *sample);
|
|
|
|
s->status &= ~(1 << 7); /* I2SOFF */
|
|
|
|
}
|
|
|
|
if (value & (1 << 4)) /* EFWR */
|
|
|
|
printf("%s: Attempt to use special function\n", __FUNCTION__);
|
2011-02-20 22:23:59 +03:00
|
|
|
s->enable = (value & 9) == 1; /* ENB && !RST*/
|
2007-04-30 05:26:42 +04:00
|
|
|
pxa2xx_i2s_update(s);
|
|
|
|
break;
|
|
|
|
case SACR1:
|
|
|
|
s->control[1] = value & 0x0039;
|
|
|
|
if (value & (1 << 5)) /* ENLBF */
|
|
|
|
printf("%s: Attempt to use loopback function\n", __FUNCTION__);
|
|
|
|
if (value & (1 << 4)) /* DPRL */
|
|
|
|
s->fifo_len = 0;
|
|
|
|
pxa2xx_i2s_update(s);
|
|
|
|
break;
|
|
|
|
case SAIMR:
|
|
|
|
s->mask = value & 0x0078;
|
|
|
|
pxa2xx_i2s_update(s);
|
|
|
|
break;
|
|
|
|
case SAICR:
|
|
|
|
s->status &= ~(value & (3 << 5));
|
|
|
|
pxa2xx_i2s_update(s);
|
|
|
|
break;
|
|
|
|
case SADIV:
|
|
|
|
s->clk = value & 0x007f;
|
|
|
|
break;
|
|
|
|
case SADR:
|
|
|
|
if (s->tx_len && s->enable) {
|
|
|
|
s->tx_len --;
|
|
|
|
pxa2xx_i2s_update(s);
|
|
|
|
s->codec_out(s->opaque, value);
|
|
|
|
} else if (s->fifo_len < 16) {
|
|
|
|
s->fifo[s->fifo_len ++] = value;
|
|
|
|
pxa2xx_i2s_update(s);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static const MemoryRegionOps pxa2xx_i2s_ops = {
|
|
|
|
.read = pxa2xx_i2s_read,
|
|
|
|
.write = pxa2xx_i2s_write,
|
|
|
|
.endianness = DEVICE_NATIVE_ENDIAN,
|
2007-04-30 05:26:42 +04:00
|
|
|
};
|
|
|
|
|
2010-12-02 16:14:56 +03:00
|
|
|
static const VMStateDescription vmstate_pxa2xx_i2s = {
|
|
|
|
.name = "pxa2xx_i2s",
|
|
|
|
.version_id = 0,
|
|
|
|
.minimum_version_id = 0,
|
|
|
|
.minimum_version_id_old = 0,
|
|
|
|
.fields = (VMStateField[]) {
|
|
|
|
VMSTATE_UINT32_ARRAY(control, PXA2xxI2SState, 2),
|
|
|
|
VMSTATE_UINT32(status, PXA2xxI2SState),
|
|
|
|
VMSTATE_UINT32(mask, PXA2xxI2SState),
|
|
|
|
VMSTATE_UINT32(clk, PXA2xxI2SState),
|
|
|
|
VMSTATE_INT32(enable, PXA2xxI2SState),
|
|
|
|
VMSTATE_INT32(rx_len, PXA2xxI2SState),
|
|
|
|
VMSTATE_INT32(tx_len, PXA2xxI2SState),
|
|
|
|
VMSTATE_INT32(fifo_len, PXA2xxI2SState),
|
|
|
|
VMSTATE_END_OF_LIST()
|
|
|
|
}
|
|
|
|
};
|
2007-05-24 22:50:09 +04:00
|
|
|
|
2007-04-30 05:26:42 +04:00
|
|
|
static void pxa2xx_i2s_data_req(void *opaque, int tx, int rx)
|
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxI2SState *s = (PXA2xxI2SState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
uint32_t *sample;
|
|
|
|
|
|
|
|
/* Signal FIFO errors */
|
|
|
|
if (s->enable && s->tx_len)
|
|
|
|
s->status |= 1 << 5; /* TUR */
|
|
|
|
if (s->enable && s->rx_len)
|
|
|
|
s->status |= 1 << 6; /* ROR */
|
|
|
|
|
|
|
|
/* Should be tx - MIN(tx, s->fifo_len) but we don't really need to
|
|
|
|
* handle the cases where it makes a difference. */
|
|
|
|
s->tx_len = tx - s->fifo_len;
|
|
|
|
s->rx_len = rx;
|
|
|
|
/* Note that is s->codec_out wasn't set, we wouldn't get called. */
|
|
|
|
if (s->enable)
|
|
|
|
for (sample = s->fifo; s->fifo_len; s->fifo_len --, sample ++)
|
|
|
|
s->codec_out(s->opaque, *sample);
|
|
|
|
pxa2xx_i2s_update(s);
|
|
|
|
}
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static PXA2xxI2SState *pxa2xx_i2s_init(MemoryRegion *sysmem,
|
|
|
|
target_phys_addr_t base,
|
2011-03-03 17:04:51 +03:00
|
|
|
qemu_irq irq, qemu_irq rx_dma, qemu_irq tx_dma)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxI2SState *s = (PXA2xxI2SState *)
|
2011-08-21 07:09:37 +04:00
|
|
|
g_malloc0(sizeof(PXA2xxI2SState));
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
s->irq = irq;
|
2011-03-03 17:04:51 +03:00
|
|
|
s->rx_dma = rx_dma;
|
|
|
|
s->tx_dma = tx_dma;
|
2007-04-30 05:26:42 +04:00
|
|
|
s->data_req = pxa2xx_i2s_data_req;
|
|
|
|
|
|
|
|
pxa2xx_i2s_reset(s);
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
memory_region_init_io(&s->iomem, &pxa2xx_i2s_ops, s,
|
|
|
|
"pxa2xx-i2s", 0x100000);
|
|
|
|
memory_region_add_subregion(sysmem, base, &s->iomem);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2010-12-02 16:14:56 +03:00
|
|
|
vmstate_register(NULL, base, &vmstate_pxa2xx_i2s, s);
|
2007-05-24 22:50:09 +04:00
|
|
|
|
2007-04-30 05:26:42 +04:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* PXA Fast Infra-red Communications Port */
|
2009-05-10 04:44:56 +04:00
|
|
|
struct PXA2xxFIrState {
|
2011-09-25 19:19:19 +04:00
|
|
|
MemoryRegion iomem;
|
2007-04-30 05:26:42 +04:00
|
|
|
qemu_irq irq;
|
2011-03-03 17:04:51 +03:00
|
|
|
qemu_irq rx_dma;
|
|
|
|
qemu_irq tx_dma;
|
2007-04-30 05:26:42 +04:00
|
|
|
int enable;
|
|
|
|
CharDriverState *chr;
|
|
|
|
|
|
|
|
uint8_t control[3];
|
|
|
|
uint8_t status[2];
|
|
|
|
|
|
|
|
int rx_len;
|
|
|
|
int rx_start;
|
|
|
|
uint8_t rx_fifo[64];
|
|
|
|
};
|
|
|
|
|
2009-05-10 04:44:56 +04:00
|
|
|
static void pxa2xx_fir_reset(PXA2xxFIrState *s)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
|
|
|
s->control[0] = 0x00;
|
|
|
|
s->control[1] = 0x00;
|
|
|
|
s->control[2] = 0x00;
|
|
|
|
s->status[0] = 0x00;
|
|
|
|
s->status[1] = 0x00;
|
|
|
|
s->enable = 0;
|
|
|
|
}
|
|
|
|
|
2009-05-10 04:44:56 +04:00
|
|
|
static inline void pxa2xx_fir_update(PXA2xxFIrState *s)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
|
|
|
static const int tresh[4] = { 8, 16, 32, 0 };
|
|
|
|
int intr = 0;
|
|
|
|
if ((s->control[0] & (1 << 4)) && /* RXE */
|
|
|
|
s->rx_len >= tresh[s->control[2] & 3]) /* TRIG */
|
|
|
|
s->status[0] |= 1 << 4; /* RFS */
|
|
|
|
else
|
|
|
|
s->status[0] &= ~(1 << 4); /* RFS */
|
|
|
|
if (s->control[0] & (1 << 3)) /* TXE */
|
|
|
|
s->status[0] |= 1 << 3; /* TFS */
|
|
|
|
else
|
|
|
|
s->status[0] &= ~(1 << 3); /* TFS */
|
|
|
|
if (s->rx_len)
|
|
|
|
s->status[1] |= 1 << 2; /* RNE */
|
|
|
|
else
|
|
|
|
s->status[1] &= ~(1 << 2); /* RNE */
|
|
|
|
if (s->control[0] & (1 << 4)) /* RXE */
|
|
|
|
s->status[1] |= 1 << 0; /* RSY */
|
|
|
|
else
|
|
|
|
s->status[1] &= ~(1 << 0); /* RSY */
|
|
|
|
|
|
|
|
intr |= (s->control[0] & (1 << 5)) && /* RIE */
|
|
|
|
(s->status[0] & (1 << 4)); /* RFS */
|
|
|
|
intr |= (s->control[0] & (1 << 6)) && /* TIE */
|
|
|
|
(s->status[0] & (1 << 3)); /* TFS */
|
|
|
|
intr |= (s->control[2] & (1 << 4)) && /* TRAIL */
|
|
|
|
(s->status[0] & (1 << 6)); /* EOC */
|
|
|
|
intr |= (s->control[0] & (1 << 2)) && /* TUS */
|
|
|
|
(s->status[0] & (1 << 1)); /* TUR */
|
|
|
|
intr |= s->status[0] & 0x25; /* FRE, RAB, EIF */
|
|
|
|
|
2011-03-03 17:04:51 +03:00
|
|
|
qemu_set_irq(s->rx_dma, (s->status[0] >> 4) & 1);
|
|
|
|
qemu_set_irq(s->tx_dma, (s->status[0] >> 3) & 1);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
qemu_set_irq(s->irq, intr && s->enable);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define ICCR0 0x00 /* FICP Control register 0 */
|
|
|
|
#define ICCR1 0x04 /* FICP Control register 1 */
|
|
|
|
#define ICCR2 0x08 /* FICP Control register 2 */
|
|
|
|
#define ICDR 0x0c /* FICP Data register */
|
|
|
|
#define ICSR0 0x14 /* FICP Status register 0 */
|
|
|
|
#define ICSR1 0x18 /* FICP Status register 1 */
|
|
|
|
#define ICFOR 0x1c /* FICP FIFO Occupancy Status register */
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static uint64_t pxa2xx_fir_read(void *opaque, target_phys_addr_t addr,
|
|
|
|
unsigned size)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxFIrState *s = (PXA2xxFIrState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
uint8_t ret;
|
|
|
|
|
|
|
|
switch (addr) {
|
|
|
|
case ICCR0:
|
|
|
|
return s->control[0];
|
|
|
|
case ICCR1:
|
|
|
|
return s->control[1];
|
|
|
|
case ICCR2:
|
|
|
|
return s->control[2];
|
|
|
|
case ICDR:
|
|
|
|
s->status[0] &= ~0x01;
|
|
|
|
s->status[1] &= ~0x72;
|
|
|
|
if (s->rx_len) {
|
|
|
|
s->rx_len --;
|
|
|
|
ret = s->rx_fifo[s->rx_start ++];
|
|
|
|
s->rx_start &= 63;
|
|
|
|
pxa2xx_fir_update(s);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
printf("%s: Rx FIFO underrun.\n", __FUNCTION__);
|
|
|
|
break;
|
|
|
|
case ICSR0:
|
|
|
|
return s->status[0];
|
|
|
|
case ICSR1:
|
|
|
|
return s->status[1] | (1 << 3); /* TNF */
|
|
|
|
case ICFOR:
|
|
|
|
return s->rx_len;
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-10-02 01:12:16 +04:00
|
|
|
static void pxa2xx_fir_write(void *opaque, target_phys_addr_t addr,
|
2011-09-25 19:19:19 +04:00
|
|
|
uint64_t value64, unsigned size)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxFIrState *s = (PXA2xxFIrState *) opaque;
|
2011-09-25 19:19:19 +04:00
|
|
|
uint32_t value = value64;
|
2007-04-30 05:26:42 +04:00
|
|
|
uint8_t ch;
|
|
|
|
|
|
|
|
switch (addr) {
|
|
|
|
case ICCR0:
|
|
|
|
s->control[0] = value;
|
|
|
|
if (!(value & (1 << 4))) /* RXE */
|
|
|
|
s->rx_len = s->rx_start = 0;
|
2010-09-18 11:01:48 +04:00
|
|
|
if (!(value & (1 << 3))) { /* TXE */
|
|
|
|
/* Nop */
|
|
|
|
}
|
2007-04-30 05:26:42 +04:00
|
|
|
s->enable = value & 1; /* ITR */
|
|
|
|
if (!s->enable)
|
|
|
|
s->status[0] = 0;
|
|
|
|
pxa2xx_fir_update(s);
|
|
|
|
break;
|
|
|
|
case ICCR1:
|
|
|
|
s->control[1] = value;
|
|
|
|
break;
|
|
|
|
case ICCR2:
|
|
|
|
s->control[2] = value & 0x3f;
|
|
|
|
pxa2xx_fir_update(s);
|
|
|
|
break;
|
|
|
|
case ICDR:
|
|
|
|
if (s->control[2] & (1 << 2)) /* TXP */
|
|
|
|
ch = value;
|
|
|
|
else
|
|
|
|
ch = ~value;
|
|
|
|
if (s->chr && s->enable && (s->control[0] & (1 << 3))) /* TXE */
|
2011-08-15 20:17:28 +04:00
|
|
|
qemu_chr_fe_write(s->chr, &ch, 1);
|
2007-04-30 05:26:42 +04:00
|
|
|
break;
|
|
|
|
case ICSR0:
|
|
|
|
s->status[0] &= ~(value & 0x66);
|
|
|
|
pxa2xx_fir_update(s);
|
|
|
|
break;
|
|
|
|
case ICFOR:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static const MemoryRegionOps pxa2xx_fir_ops = {
|
|
|
|
.read = pxa2xx_fir_read,
|
|
|
|
.write = pxa2xx_fir_write,
|
|
|
|
.endianness = DEVICE_NATIVE_ENDIAN,
|
2007-04-30 05:26:42 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
static int pxa2xx_fir_is_empty(void *opaque)
|
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxFIrState *s = (PXA2xxFIrState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
return (s->rx_len < 64);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void pxa2xx_fir_rx(void *opaque, const uint8_t *buf, int size)
|
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxFIrState *s = (PXA2xxFIrState *) opaque;
|
2007-04-30 05:26:42 +04:00
|
|
|
if (!(s->control[0] & (1 << 4))) /* RXE */
|
|
|
|
return;
|
|
|
|
|
|
|
|
while (size --) {
|
|
|
|
s->status[1] |= 1 << 4; /* EOF */
|
|
|
|
if (s->rx_len >= 64) {
|
|
|
|
s->status[1] |= 1 << 6; /* ROR */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (s->control[2] & (1 << 3)) /* RXP */
|
|
|
|
s->rx_fifo[(s->rx_start + s->rx_len ++) & 63] = *(buf ++);
|
|
|
|
else
|
|
|
|
s->rx_fifo[(s->rx_start + s->rx_len ++) & 63] = ~*(buf ++);
|
|
|
|
}
|
|
|
|
|
|
|
|
pxa2xx_fir_update(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void pxa2xx_fir_event(void *opaque, int event)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2007-05-24 22:50:09 +04:00
|
|
|
static void pxa2xx_fir_save(QEMUFile *f, void *opaque)
|
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxFIrState *s = (PXA2xxFIrState *) opaque;
|
2007-05-24 22:50:09 +04:00
|
|
|
int i;
|
|
|
|
|
|
|
|
qemu_put_be32(f, s->enable);
|
|
|
|
|
|
|
|
qemu_put_8s(f, &s->control[0]);
|
|
|
|
qemu_put_8s(f, &s->control[1]);
|
|
|
|
qemu_put_8s(f, &s->control[2]);
|
|
|
|
qemu_put_8s(f, &s->status[0]);
|
|
|
|
qemu_put_8s(f, &s->status[1]);
|
|
|
|
|
|
|
|
qemu_put_byte(f, s->rx_len);
|
|
|
|
for (i = 0; i < s->rx_len; i ++)
|
|
|
|
qemu_put_byte(f, s->rx_fifo[(s->rx_start + i) & 63]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int pxa2xx_fir_load(QEMUFile *f, void *opaque, int version_id)
|
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxFIrState *s = (PXA2xxFIrState *) opaque;
|
2007-05-24 22:50:09 +04:00
|
|
|
int i;
|
|
|
|
|
|
|
|
s->enable = qemu_get_be32(f);
|
|
|
|
|
|
|
|
qemu_get_8s(f, &s->control[0]);
|
|
|
|
qemu_get_8s(f, &s->control[1]);
|
|
|
|
qemu_get_8s(f, &s->control[2]);
|
|
|
|
qemu_get_8s(f, &s->status[0]);
|
|
|
|
qemu_get_8s(f, &s->status[1]);
|
|
|
|
|
|
|
|
s->rx_len = qemu_get_byte(f);
|
|
|
|
s->rx_start = 0;
|
|
|
|
for (i = 0; i < s->rx_len; i ++)
|
|
|
|
s->rx_fifo[i] = qemu_get_byte(f);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
static PXA2xxFIrState *pxa2xx_fir_init(MemoryRegion *sysmem,
|
|
|
|
target_phys_addr_t base,
|
2011-03-03 17:04:51 +03:00
|
|
|
qemu_irq irq, qemu_irq rx_dma, qemu_irq tx_dma,
|
2007-04-30 05:26:42 +04:00
|
|
|
CharDriverState *chr)
|
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxFIrState *s = (PXA2xxFIrState *)
|
2011-08-21 07:09:37 +04:00
|
|
|
g_malloc0(sizeof(PXA2xxFIrState));
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
s->irq = irq;
|
2011-03-03 17:04:51 +03:00
|
|
|
s->rx_dma = rx_dma;
|
|
|
|
s->tx_dma = tx_dma;
|
2007-04-30 05:26:42 +04:00
|
|
|
s->chr = chr;
|
|
|
|
|
|
|
|
pxa2xx_fir_reset(s);
|
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
memory_region_init_io(&s->iomem, &pxa2xx_fir_ops, s, "pxa2xx-fir", 0x1000);
|
|
|
|
memory_region_add_subregion(sysmem, base, &s->iomem);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
if (chr)
|
|
|
|
qemu_chr_add_handlers(chr, pxa2xx_fir_is_empty,
|
|
|
|
pxa2xx_fir_rx, pxa2xx_fir_event, s);
|
|
|
|
|
2010-06-25 21:09:07 +04:00
|
|
|
register_savevm(NULL, "pxa2xx_fir", 0, 0, pxa2xx_fir_save,
|
|
|
|
pxa2xx_fir_load, s);
|
2007-05-24 22:50:09 +04:00
|
|
|
|
2007-04-30 05:26:42 +04:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2007-11-17 17:07:13 +03:00
|
|
|
static void pxa2xx_reset(void *opaque, int line, int level)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxState *s = (PXA2xxState *) opaque;
|
2007-11-17 17:07:13 +03:00
|
|
|
|
2007-04-30 05:26:42 +04:00
|
|
|
if (level && (s->pm_regs[PCFR >> 2] & 0x10)) { /* GPR_EN */
|
2012-05-04 01:47:04 +04:00
|
|
|
cpu_reset(CPU(s->cpu));
|
2007-04-30 05:26:42 +04:00
|
|
|
/* TODO: reset peripherals */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Initialise a PXA270 integrated chip (ARM based core). */
|
2011-08-12 03:07:19 +04:00
|
|
|
PXA2xxState *pxa270_init(MemoryRegion *address_space,
|
|
|
|
unsigned int sdram_size, const char *revision)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxState *s;
|
2011-09-25 19:19:19 +04:00
|
|
|
int i;
|
2009-07-22 18:42:57 +04:00
|
|
|
DriveInfo *dinfo;
|
2011-08-21 07:09:37 +04:00
|
|
|
s = (PXA2xxState *) g_malloc0(sizeof(PXA2xxState));
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2007-05-01 05:03:32 +04:00
|
|
|
if (revision && strncmp(revision, "pxa27", 5)) {
|
|
|
|
fprintf(stderr, "Machine requires a PXA27x processor.\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
2007-11-10 18:15:54 +03:00
|
|
|
if (!revision)
|
|
|
|
revision = "pxa270";
|
|
|
|
|
2012-05-04 01:47:04 +04:00
|
|
|
s->cpu = cpu_arm_init(revision);
|
|
|
|
if (s->cpu == NULL) {
|
2007-11-10 18:15:54 +03:00
|
|
|
fprintf(stderr, "Unable to find CPU definition\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
2007-11-17 17:07:13 +03:00
|
|
|
s->reset = qemu_allocate_irqs(pxa2xx_reset, s, 1)[0];
|
|
|
|
|
2007-05-08 23:03:12 +04:00
|
|
|
/* SDRAM & Internal Memory Storage */
|
2011-12-20 17:59:12 +04:00
|
|
|
memory_region_init_ram(&s->sdram, "pxa270.sdram", sdram_size);
|
|
|
|
vmstate_register_ram_global(&s->sdram);
|
2011-09-25 19:19:19 +04:00
|
|
|
memory_region_add_subregion(address_space, PXA2XX_SDRAM_BASE, &s->sdram);
|
2011-12-20 17:59:12 +04:00
|
|
|
memory_region_init_ram(&s->internal, "pxa270.internal", 0x40000);
|
|
|
|
vmstate_register_ram_global(&s->internal);
|
2011-09-25 19:19:19 +04:00
|
|
|
memory_region_add_subregion(address_space, PXA2XX_INTERNAL_BASE,
|
|
|
|
&s->internal);
|
2007-05-08 23:03:12 +04:00
|
|
|
|
2012-05-04 02:01:27 +04:00
|
|
|
s->pic = pxa2xx_pic_init(0x40d00000, s->cpu);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2011-02-25 14:13:38 +03:00
|
|
|
s->dma = pxa27x_dma_init(0x40000000,
|
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_DMA));
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2011-03-03 16:14:44 +03:00
|
|
|
sysbus_create_varargs("pxa27x-timer", 0x40a00000,
|
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_OST_0 + 0),
|
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_OST_0 + 1),
|
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_OST_0 + 2),
|
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_OST_0 + 3),
|
|
|
|
qdev_get_gpio_in(s->pic, PXA27X_PIC_OST_4_11),
|
|
|
|
NULL);
|
2007-04-30 05:48:07 +04:00
|
|
|
|
2012-05-04 01:47:04 +04:00
|
|
|
s->gpio = pxa2xx_gpio_init(0x40e00000, &s->cpu->env, s->pic, 121);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2009-07-22 18:42:57 +04:00
|
|
|
dinfo = drive_get(IF_SD, 0, 0);
|
|
|
|
if (!dinfo) {
|
2007-12-02 07:51:10 +03:00
|
|
|
fprintf(stderr, "qemu: missing SecureDigital device\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
2011-10-30 17:50:18 +04:00
|
|
|
s->mmc = pxa2xx_mmci_init(address_space, 0x41100000, dinfo->bdrv,
|
2011-03-03 17:04:51 +03:00
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_MMC),
|
|
|
|
qdev_get_gpio_in(s->dma, PXA2XX_RX_RQ_MMCI),
|
|
|
|
qdev_get_gpio_in(s->dma, PXA2XX_TX_RQ_MMCI));
|
2007-04-30 05:48:07 +04:00
|
|
|
|
2011-08-12 03:07:14 +04:00
|
|
|
for (i = 0; pxa270_serial[i].io_base; i++) {
|
|
|
|
if (serial_hds[i]) {
|
2011-08-12 03:07:19 +04:00
|
|
|
serial_mm_init(address_space, pxa270_serial[i].io_base, 2,
|
2011-08-12 03:07:14 +04:00
|
|
|
qdev_get_gpio_in(s->pic, pxa270_serial[i].irqn),
|
2011-08-12 03:07:15 +04:00
|
|
|
14857000 / 16, serial_hds[i],
|
2011-08-12 03:07:14 +04:00
|
|
|
DEVICE_NATIVE_ENDIAN);
|
|
|
|
} else {
|
2007-04-30 05:26:42 +04:00
|
|
|
break;
|
2011-08-12 03:07:14 +04:00
|
|
|
}
|
|
|
|
}
|
2007-04-30 05:26:42 +04:00
|
|
|
if (serial_hds[i])
|
2011-09-25 19:19:19 +04:00
|
|
|
s->fir = pxa2xx_fir_init(address_space, 0x40800000,
|
2011-02-25 14:13:38 +03:00
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_ICP),
|
2011-03-03 17:04:51 +03:00
|
|
|
qdev_get_gpio_in(s->dma, PXA2XX_RX_RQ_ICP),
|
|
|
|
qdev_get_gpio_in(s->dma, PXA2XX_TX_RQ_ICP),
|
|
|
|
serial_hds[i]);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2011-10-30 17:50:19 +04:00
|
|
|
s->lcd = pxa2xx_lcdc_init(address_space, 0x44000000,
|
2011-02-25 14:13:38 +03:00
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_LCD));
|
2007-04-30 05:48:07 +04:00
|
|
|
|
2007-04-30 05:26:42 +04:00
|
|
|
s->cm_base = 0x41300000;
|
2007-07-24 05:07:44 +04:00
|
|
|
s->cm_regs[CCCR >> 2] = 0x02000210; /* 416.0 MHz */
|
2007-04-30 05:26:42 +04:00
|
|
|
s->clkcfg = 0x00000009; /* Turbo mode active */
|
2011-09-25 19:19:19 +04:00
|
|
|
memory_region_init_io(&s->cm_iomem, &pxa2xx_cm_ops, s, "pxa2xx-cm", 0x1000);
|
|
|
|
memory_region_add_subregion(address_space, s->cm_base, &s->cm_iomem);
|
2010-12-02 16:31:14 +03:00
|
|
|
vmstate_register(NULL, 0, &vmstate_pxa2xx_cm, s);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2012-05-04 01:47:04 +04:00
|
|
|
cpu_arm_set_cp_io(&s->cpu->env, 14, pxa2xx_cp14_read, pxa2xx_cp14_write, s);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
s->mm_base = 0x48000000;
|
|
|
|
s->mm_regs[MDMRS >> 2] = 0x00020002;
|
|
|
|
s->mm_regs[MDREFR >> 2] = 0x03ca4000;
|
|
|
|
s->mm_regs[MECR >> 2] = 0x00000001; /* Two PC Card sockets */
|
2011-09-25 19:19:19 +04:00
|
|
|
memory_region_init_io(&s->mm_iomem, &pxa2xx_mm_ops, s, "pxa2xx-mm", 0x1000);
|
|
|
|
memory_region_add_subregion(address_space, s->mm_base, &s->mm_iomem);
|
2010-12-02 16:36:57 +03:00
|
|
|
vmstate_register(NULL, 0, &vmstate_pxa2xx_mm, s);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2007-05-28 15:26:15 +04:00
|
|
|
s->pm_base = 0x40f00000;
|
2011-09-25 19:19:19 +04:00
|
|
|
memory_region_init_io(&s->pm_iomem, &pxa2xx_pm_ops, s, "pxa2xx-pm", 0x100);
|
|
|
|
memory_region_add_subregion(address_space, s->pm_base, &s->pm_iomem);
|
2010-12-02 16:54:38 +03:00
|
|
|
vmstate_register(NULL, 0, &vmstate_pxa2xx_pm, s);
|
2007-05-28 15:26:15 +04:00
|
|
|
|
2007-04-30 05:26:42 +04:00
|
|
|
for (i = 0; pxa27x_ssp[i].io_base; i ++);
|
2011-08-21 07:09:37 +04:00
|
|
|
s->ssp = (SSIBus **)g_malloc0(sizeof(SSIBus *) * i);
|
2007-04-30 05:26:42 +04:00
|
|
|
for (i = 0; pxa27x_ssp[i].io_base; i ++) {
|
2009-05-15 01:35:09 +04:00
|
|
|
DeviceState *dev;
|
|
|
|
dev = sysbus_create_simple("pxa2xx-ssp", pxa27x_ssp[i].io_base,
|
2011-02-25 14:13:38 +03:00
|
|
|
qdev_get_gpio_in(s->pic, pxa27x_ssp[i].irqn));
|
2009-05-23 03:05:19 +04:00
|
|
|
s->ssp[i] = (SSIBus *)qdev_get_child_bus(dev, "ssi");
|
2007-04-30 05:26:42 +04:00
|
|
|
}
|
|
|
|
|
2007-04-30 05:48:07 +04:00
|
|
|
if (usb_enabled) {
|
2010-04-05 22:57:40 +04:00
|
|
|
sysbus_create_simple("sysbus-ohci", 0x4c000000,
|
2011-02-25 14:13:38 +03:00
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_USBH1));
|
2007-04-30 05:48:07 +04:00
|
|
|
}
|
|
|
|
|
2011-10-30 17:50:12 +04:00
|
|
|
s->pcmcia[0] = pxa2xx_pcmcia_init(address_space, 0x20000000);
|
|
|
|
s->pcmcia[1] = pxa2xx_pcmcia_init(address_space, 0x30000000);
|
2007-04-30 05:48:07 +04:00
|
|
|
|
2011-03-03 17:13:42 +03:00
|
|
|
sysbus_create_simple("pxa2xx_rtc", 0x40900000,
|
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_RTCALARM));
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2011-02-25 14:13:38 +03:00
|
|
|
s->i2c[0] = pxa2xx_i2c_init(0x40301600,
|
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_I2C), 0xffff);
|
|
|
|
s->i2c[1] = pxa2xx_i2c_init(0x40f00100,
|
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_PWRI2C), 0xff);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
s->i2s = pxa2xx_i2s_init(address_space, 0x40400000,
|
2011-03-03 17:04:51 +03:00
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_I2S),
|
|
|
|
qdev_get_gpio_in(s->dma, PXA2XX_RX_RQ_I2S),
|
|
|
|
qdev_get_gpio_in(s->dma, PXA2XX_TX_RQ_I2S));
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2011-10-30 17:50:15 +04:00
|
|
|
s->kp = pxa27x_keypad_init(address_space, 0x41500000,
|
2011-02-25 14:13:38 +03:00
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_KEYPAD));
|
2007-12-16 15:13:51 +03:00
|
|
|
|
2007-04-30 05:26:42 +04:00
|
|
|
/* GPIO1 resets the processor */
|
2007-07-12 14:59:21 +04:00
|
|
|
/* The handler can be overridden by board-specific code */
|
2011-01-21 19:57:50 +03:00
|
|
|
qdev_connect_gpio_out(s->gpio, 1, s->reset);
|
2007-04-30 05:26:42 +04:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Initialise a PXA255 integrated chip (ARM based core). */
|
2011-08-12 03:07:19 +04:00
|
|
|
PXA2xxState *pxa255_init(MemoryRegion *address_space, unsigned int sdram_size)
|
2007-04-30 05:26:42 +04:00
|
|
|
{
|
2009-05-10 04:44:56 +04:00
|
|
|
PXA2xxState *s;
|
2011-09-25 19:19:19 +04:00
|
|
|
int i;
|
2009-07-22 18:42:57 +04:00
|
|
|
DriveInfo *dinfo;
|
2007-11-10 18:15:54 +03:00
|
|
|
|
2011-08-21 07:09:37 +04:00
|
|
|
s = (PXA2xxState *) g_malloc0(sizeof(PXA2xxState));
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2012-05-04 01:47:04 +04:00
|
|
|
s->cpu = cpu_arm_init("pxa255");
|
|
|
|
if (s->cpu == NULL) {
|
2007-11-10 18:15:54 +03:00
|
|
|
fprintf(stderr, "Unable to find CPU definition\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
2007-11-17 17:07:13 +03:00
|
|
|
s->reset = qemu_allocate_irqs(pxa2xx_reset, s, 1)[0];
|
|
|
|
|
2007-05-08 23:03:12 +04:00
|
|
|
/* SDRAM & Internal Memory Storage */
|
2011-12-20 17:59:12 +04:00
|
|
|
memory_region_init_ram(&s->sdram, "pxa255.sdram", sdram_size);
|
|
|
|
vmstate_register_ram_global(&s->sdram);
|
2011-09-25 19:19:19 +04:00
|
|
|
memory_region_add_subregion(address_space, PXA2XX_SDRAM_BASE, &s->sdram);
|
2011-12-20 17:59:12 +04:00
|
|
|
memory_region_init_ram(&s->internal, "pxa255.internal",
|
2011-09-25 19:19:19 +04:00
|
|
|
PXA2XX_INTERNAL_SIZE);
|
2011-12-20 17:59:12 +04:00
|
|
|
vmstate_register_ram_global(&s->internal);
|
2011-09-25 19:19:19 +04:00
|
|
|
memory_region_add_subregion(address_space, PXA2XX_INTERNAL_BASE,
|
|
|
|
&s->internal);
|
2007-05-08 23:03:12 +04:00
|
|
|
|
2012-05-04 02:01:27 +04:00
|
|
|
s->pic = pxa2xx_pic_init(0x40d00000, s->cpu);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2011-02-25 14:13:38 +03:00
|
|
|
s->dma = pxa255_dma_init(0x40000000,
|
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_DMA));
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2011-03-03 16:14:44 +03:00
|
|
|
sysbus_create_varargs("pxa25x-timer", 0x40a00000,
|
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_OST_0 + 0),
|
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_OST_0 + 1),
|
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_OST_0 + 2),
|
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_OST_0 + 3),
|
|
|
|
NULL);
|
2007-04-30 05:48:07 +04:00
|
|
|
|
2012-05-04 01:47:04 +04:00
|
|
|
s->gpio = pxa2xx_gpio_init(0x40e00000, &s->cpu->env, s->pic, 85);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2009-07-22 18:42:57 +04:00
|
|
|
dinfo = drive_get(IF_SD, 0, 0);
|
|
|
|
if (!dinfo) {
|
2007-12-02 07:51:10 +03:00
|
|
|
fprintf(stderr, "qemu: missing SecureDigital device\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
2011-10-30 17:50:18 +04:00
|
|
|
s->mmc = pxa2xx_mmci_init(address_space, 0x41100000, dinfo->bdrv,
|
2011-03-03 17:04:51 +03:00
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_MMC),
|
|
|
|
qdev_get_gpio_in(s->dma, PXA2XX_RX_RQ_MMCI),
|
|
|
|
qdev_get_gpio_in(s->dma, PXA2XX_TX_RQ_MMCI));
|
2007-04-30 05:48:07 +04:00
|
|
|
|
2011-08-12 03:07:14 +04:00
|
|
|
for (i = 0; pxa255_serial[i].io_base; i++) {
|
2010-03-21 22:47:11 +03:00
|
|
|
if (serial_hds[i]) {
|
2011-08-12 03:07:19 +04:00
|
|
|
serial_mm_init(address_space, pxa255_serial[i].io_base, 2,
|
2011-08-12 03:07:14 +04:00
|
|
|
qdev_get_gpio_in(s->pic, pxa255_serial[i].irqn),
|
2011-08-12 03:07:15 +04:00
|
|
|
14745600 / 16, serial_hds[i],
|
2011-08-12 03:07:14 +04:00
|
|
|
DEVICE_NATIVE_ENDIAN);
|
2010-03-21 22:47:11 +03:00
|
|
|
} else {
|
2007-04-30 05:26:42 +04:00
|
|
|
break;
|
2010-03-21 22:47:11 +03:00
|
|
|
}
|
2011-08-12 03:07:14 +04:00
|
|
|
}
|
2007-04-30 05:26:42 +04:00
|
|
|
if (serial_hds[i])
|
2011-09-25 19:19:19 +04:00
|
|
|
s->fir = pxa2xx_fir_init(address_space, 0x40800000,
|
2011-02-25 14:13:38 +03:00
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_ICP),
|
2011-03-03 17:04:51 +03:00
|
|
|
qdev_get_gpio_in(s->dma, PXA2XX_RX_RQ_ICP),
|
|
|
|
qdev_get_gpio_in(s->dma, PXA2XX_TX_RQ_ICP),
|
|
|
|
serial_hds[i]);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2011-10-30 17:50:19 +04:00
|
|
|
s->lcd = pxa2xx_lcdc_init(address_space, 0x44000000,
|
2011-02-25 14:13:38 +03:00
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_LCD));
|
2007-04-30 05:48:07 +04:00
|
|
|
|
2007-04-30 05:26:42 +04:00
|
|
|
s->cm_base = 0x41300000;
|
2007-07-24 05:07:44 +04:00
|
|
|
s->cm_regs[CCCR >> 2] = 0x02000210; /* 416.0 MHz */
|
2007-04-30 05:26:42 +04:00
|
|
|
s->clkcfg = 0x00000009; /* Turbo mode active */
|
2011-09-25 19:19:19 +04:00
|
|
|
memory_region_init_io(&s->cm_iomem, &pxa2xx_cm_ops, s, "pxa2xx-cm", 0x1000);
|
|
|
|
memory_region_add_subregion(address_space, s->cm_base, &s->cm_iomem);
|
2010-12-02 16:31:14 +03:00
|
|
|
vmstate_register(NULL, 0, &vmstate_pxa2xx_cm, s);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2012-05-04 01:47:04 +04:00
|
|
|
cpu_arm_set_cp_io(&s->cpu->env, 14, pxa2xx_cp14_read, pxa2xx_cp14_write, s);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
s->mm_base = 0x48000000;
|
|
|
|
s->mm_regs[MDMRS >> 2] = 0x00020002;
|
|
|
|
s->mm_regs[MDREFR >> 2] = 0x03ca4000;
|
|
|
|
s->mm_regs[MECR >> 2] = 0x00000001; /* Two PC Card sockets */
|
2011-09-25 19:19:19 +04:00
|
|
|
memory_region_init_io(&s->mm_iomem, &pxa2xx_mm_ops, s, "pxa2xx-mm", 0x1000);
|
|
|
|
memory_region_add_subregion(address_space, s->mm_base, &s->mm_iomem);
|
2010-12-02 16:36:57 +03:00
|
|
|
vmstate_register(NULL, 0, &vmstate_pxa2xx_mm, s);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2007-05-28 15:26:15 +04:00
|
|
|
s->pm_base = 0x40f00000;
|
2011-09-25 19:19:19 +04:00
|
|
|
memory_region_init_io(&s->pm_iomem, &pxa2xx_pm_ops, s, "pxa2xx-pm", 0x100);
|
|
|
|
memory_region_add_subregion(address_space, s->pm_base, &s->pm_iomem);
|
2010-12-02 16:54:38 +03:00
|
|
|
vmstate_register(NULL, 0, &vmstate_pxa2xx_pm, s);
|
2007-05-28 15:26:15 +04:00
|
|
|
|
2007-04-30 05:26:42 +04:00
|
|
|
for (i = 0; pxa255_ssp[i].io_base; i ++);
|
2011-08-21 07:09:37 +04:00
|
|
|
s->ssp = (SSIBus **)g_malloc0(sizeof(SSIBus *) * i);
|
2007-04-30 05:26:42 +04:00
|
|
|
for (i = 0; pxa255_ssp[i].io_base; i ++) {
|
2009-05-15 01:35:09 +04:00
|
|
|
DeviceState *dev;
|
|
|
|
dev = sysbus_create_simple("pxa2xx-ssp", pxa255_ssp[i].io_base,
|
2011-02-25 14:13:38 +03:00
|
|
|
qdev_get_gpio_in(s->pic, pxa255_ssp[i].irqn));
|
2009-05-23 03:05:19 +04:00
|
|
|
s->ssp[i] = (SSIBus *)qdev_get_child_bus(dev, "ssi");
|
2007-04-30 05:26:42 +04:00
|
|
|
}
|
|
|
|
|
2007-04-30 05:48:07 +04:00
|
|
|
if (usb_enabled) {
|
2010-04-05 22:57:40 +04:00
|
|
|
sysbus_create_simple("sysbus-ohci", 0x4c000000,
|
2011-02-25 14:13:38 +03:00
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_USBH1));
|
2007-04-30 05:48:07 +04:00
|
|
|
}
|
|
|
|
|
2011-10-30 17:50:12 +04:00
|
|
|
s->pcmcia[0] = pxa2xx_pcmcia_init(address_space, 0x20000000);
|
|
|
|
s->pcmcia[1] = pxa2xx_pcmcia_init(address_space, 0x30000000);
|
2007-04-30 05:48:07 +04:00
|
|
|
|
2011-03-03 17:13:42 +03:00
|
|
|
sysbus_create_simple("pxa2xx_rtc", 0x40900000,
|
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_RTCALARM));
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2011-02-25 14:13:38 +03:00
|
|
|
s->i2c[0] = pxa2xx_i2c_init(0x40301600,
|
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_I2C), 0xffff);
|
|
|
|
s->i2c[1] = pxa2xx_i2c_init(0x40f00100,
|
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_PWRI2C), 0xff);
|
2007-04-30 05:26:42 +04:00
|
|
|
|
2011-09-25 19:19:19 +04:00
|
|
|
s->i2s = pxa2xx_i2s_init(address_space, 0x40400000,
|
2011-03-03 17:04:51 +03:00
|
|
|
qdev_get_gpio_in(s->pic, PXA2XX_PIC_I2S),
|
|
|
|
qdev_get_gpio_in(s->dma, PXA2XX_RX_RQ_I2S),
|
|
|
|
qdev_get_gpio_in(s->dma, PXA2XX_TX_RQ_I2S));
|
2007-04-30 05:26:42 +04:00
|
|
|
|
|
|
|
/* GPIO1 resets the processor */
|
2007-07-12 14:59:21 +04:00
|
|
|
/* The handler can be overridden by board-specific code */
|
2011-01-21 19:57:50 +03:00
|
|
|
qdev_connect_gpio_out(s->gpio, 1, s->reset);
|
2007-04-30 05:26:42 +04:00
|
|
|
return s;
|
|
|
|
}
|
2009-05-15 01:35:08 +04:00
|
|
|
|
2012-01-24 23:12:29 +04:00
|
|
|
static void pxa2xx_ssp_class_init(ObjectClass *klass, void *data)
|
|
|
|
{
|
|
|
|
SysBusDeviceClass *sdc = SYS_BUS_DEVICE_CLASS(klass);
|
|
|
|
|
|
|
|
sdc->init = pxa2xx_ssp_init;
|
|
|
|
}
|
|
|
|
|
2011-12-08 07:34:16 +04:00
|
|
|
static TypeInfo pxa2xx_ssp_info = {
|
|
|
|
.name = "pxa2xx-ssp",
|
|
|
|
.parent = TYPE_SYS_BUS_DEVICE,
|
|
|
|
.instance_size = sizeof(PXA2xxSSPState),
|
|
|
|
.class_init = pxa2xx_ssp_class_init,
|
2012-01-24 23:12:29 +04:00
|
|
|
};
|
|
|
|
|
2012-02-09 18:20:55 +04:00
|
|
|
static void pxa2xx_register_types(void)
|
2009-05-15 01:35:08 +04:00
|
|
|
{
|
2011-12-08 07:34:16 +04:00
|
|
|
type_register_static(&pxa2xx_i2c_slave_info);
|
|
|
|
type_register_static(&pxa2xx_ssp_info);
|
|
|
|
type_register_static(&pxa2xx_i2c_info);
|
|
|
|
type_register_static(&pxa2xx_rtc_sysbus_info);
|
2009-05-15 01:35:08 +04:00
|
|
|
}
|
|
|
|
|
2012-02-09 18:20:55 +04:00
|
|
|
type_init(pxa2xx_register_types)
|