Commit c1713132e07955819477a87a0ce830358e77a147
1 parent
201a51fc
Core features of ARM XScale processors. Main PXA270 and PXA255 peripherals.
git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@2749 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
11 changed files
with
3127 additions
and
16 deletions
hw/pxa.h
0 → 100644
1 | +/* | |
2 | + * Intel XScale PXA255/270 processor support. | |
3 | + * | |
4 | + * Copyright (c) 2006 Openedhand Ltd. | |
5 | + * Written by Andrzej Zaborowski <balrog@zabor.org> | |
6 | + * | |
7 | + * This code is licenced under the GPL. | |
8 | + */ | |
9 | +#ifndef PXA_H | |
10 | +# define PXA_H "pxa.h" | |
11 | + | |
12 | +/* Interrupt numbers */ | |
13 | +# define PXA2XX_PIC_SSP3 0 | |
14 | +# define PXA2XX_PIC_USBH2 2 | |
15 | +# define PXA2XX_PIC_USBH1 3 | |
16 | +# define PXA2XX_PIC_PWRI2C 6 | |
17 | +# define PXA25X_PIC_HWUART 7 | |
18 | +# define PXA27X_PIC_OST_4_11 7 | |
19 | +# define PXA2XX_PIC_GPIO_0 8 | |
20 | +# define PXA2XX_PIC_GPIO_1 9 | |
21 | +# define PXA2XX_PIC_GPIO_X 10 | |
22 | +# define PXA2XX_PIC_I2S 13 | |
23 | +# define PXA26X_PIC_ASSP 15 | |
24 | +# define PXA25X_PIC_NSSP 16 | |
25 | +# define PXA27X_PIC_SSP2 16 | |
26 | +# define PXA2XX_PIC_LCD 17 | |
27 | +# define PXA2XX_PIC_I2C 18 | |
28 | +# define PXA2XX_PIC_ICP 19 | |
29 | +# define PXA2XX_PIC_STUART 20 | |
30 | +# define PXA2XX_PIC_BTUART 21 | |
31 | +# define PXA2XX_PIC_FFUART 22 | |
32 | +# define PXA2XX_PIC_MMC 23 | |
33 | +# define PXA2XX_PIC_SSP 24 | |
34 | +# define PXA2XX_PIC_DMA 25 | |
35 | +# define PXA2XX_PIC_OST_0 26 | |
36 | +# define PXA2XX_PIC_RTC1HZ 30 | |
37 | +# define PXA2XX_PIC_RTCALARM 31 | |
38 | + | |
39 | +/* DMA requests */ | |
40 | +# define PXA2XX_RX_RQ_I2S 2 | |
41 | +# define PXA2XX_TX_RQ_I2S 3 | |
42 | +# define PXA2XX_RX_RQ_BTUART 4 | |
43 | +# define PXA2XX_TX_RQ_BTUART 5 | |
44 | +# define PXA2XX_RX_RQ_FFUART 6 | |
45 | +# define PXA2XX_TX_RQ_FFUART 7 | |
46 | +# define PXA2XX_RX_RQ_SSP1 13 | |
47 | +# define PXA2XX_TX_RQ_SSP1 14 | |
48 | +# define PXA2XX_RX_RQ_SSP2 15 | |
49 | +# define PXA2XX_TX_RQ_SSP2 16 | |
50 | +# define PXA2XX_RX_RQ_ICP 17 | |
51 | +# define PXA2XX_TX_RQ_ICP 18 | |
52 | +# define PXA2XX_RX_RQ_STUART 19 | |
53 | +# define PXA2XX_TX_RQ_STUART 20 | |
54 | +# define PXA2XX_RX_RQ_MMCI 21 | |
55 | +# define PXA2XX_TX_RQ_MMCI 22 | |
56 | +# define PXA2XX_USB_RQ(x) ((x) + 24) | |
57 | +# define PXA2XX_RX_RQ_SSP3 66 | |
58 | +# define PXA2XX_TX_RQ_SSP3 67 | |
59 | + | |
60 | +# define PXA2XX_RAM_BASE 0xa0000000 | |
61 | + | |
62 | +/* pxa2xx_pic.c */ | |
63 | +struct pxa2xx_pic_state_s; | |
64 | +qemu_irq *pxa2xx_pic_init(target_phys_addr_t base, CPUState *env); | |
65 | + | |
66 | +/* pxa2xx_gpio.c */ | |
67 | +struct pxa2xx_gpio_info_s; | |
68 | +struct pxa2xx_gpio_info_s *pxa2xx_gpio_init(target_phys_addr_t base, | |
69 | + CPUState *env, qemu_irq *pic, int lines); | |
70 | +void pxa2xx_gpio_set(struct pxa2xx_gpio_info_s *s, int line, int level); | |
71 | +void pxa2xx_gpio_handler_set(struct pxa2xx_gpio_info_s *s, int line, | |
72 | + gpio_handler_t handler, void *opaque); | |
73 | +void pxa2xx_gpio_read_notifier(struct pxa2xx_gpio_info_s *s, | |
74 | + void (*handler)(void *opaque), void *opaque); | |
75 | + | |
76 | +/* pxa2xx_dma.c */ | |
77 | +struct pxa2xx_dma_state_s; | |
78 | +struct pxa2xx_dma_state_s *pxa255_dma_init(target_phys_addr_t base, | |
79 | + qemu_irq irq); | |
80 | +struct pxa2xx_dma_state_s *pxa27x_dma_init(target_phys_addr_t base, | |
81 | + qemu_irq irq); | |
82 | +void pxa2xx_dma_request(struct pxa2xx_dma_state_s *s, int req_num, int on); | |
83 | + | |
84 | +/* pxa2xx.c */ | |
85 | +struct pxa2xx_ssp_s; | |
86 | +void pxa2xx_ssp_attach(struct pxa2xx_ssp_s *port, | |
87 | + uint32_t (*readfn)(void *opaque), | |
88 | + void (*writefn)(void *opaque, uint32_t value), void *opaque); | |
89 | + | |
90 | +struct pxa2xx_i2s_s; | |
91 | +struct pxa2xx_fir_s; | |
92 | + | |
93 | +struct pxa2xx_state_s { | |
94 | + CPUState *env; | |
95 | + qemu_irq *pic; | |
96 | + struct pxa2xx_dma_state_s *dma; | |
97 | + struct pxa2xx_gpio_info_s *gpio; | |
98 | + struct pxa2xx_ssp_s **ssp; | |
99 | + struct pxa2xx_i2s_s *i2s; | |
100 | + struct pxa2xx_fir_s *fir; | |
101 | + | |
102 | + /* Power management */ | |
103 | + target_phys_addr_t pm_base; | |
104 | + uint32_t pm_regs[0x40]; | |
105 | + | |
106 | + /* Clock management */ | |
107 | + target_phys_addr_t cm_base; | |
108 | + uint32_t cm_regs[4]; | |
109 | + uint32_t clkcfg; | |
110 | + | |
111 | + /* Memory management */ | |
112 | + target_phys_addr_t mm_base; | |
113 | + uint32_t mm_regs[0x1a]; | |
114 | + | |
115 | + /* Performance monitoring */ | |
116 | + uint32_t pmnc; | |
117 | + | |
118 | + /* Real-Time clock */ | |
119 | + target_phys_addr_t rtc_base; | |
120 | + uint32_t rttr; | |
121 | + uint32_t rtsr; | |
122 | + uint32_t rtar; | |
123 | + uint32_t rdar1; | |
124 | + uint32_t rdar2; | |
125 | + uint32_t ryar1; | |
126 | + uint32_t ryar2; | |
127 | + uint32_t swar1; | |
128 | + uint32_t swar2; | |
129 | + uint32_t piar; | |
130 | + uint32_t last_rcnr; | |
131 | + uint32_t last_rdcr; | |
132 | + uint32_t last_rycr; | |
133 | + uint32_t last_swcr; | |
134 | + uint32_t last_rtcpicr; | |
135 | + int64_t last_hz; | |
136 | + int64_t last_sw; | |
137 | + int64_t last_pi; | |
138 | + QEMUTimer *rtc_hz; | |
139 | + QEMUTimer *rtc_rdal1; | |
140 | + QEMUTimer *rtc_rdal2; | |
141 | + QEMUTimer *rtc_swal1; | |
142 | + QEMUTimer *rtc_swal2; | |
143 | + QEMUTimer *rtc_pi; | |
144 | +}; | |
145 | + | |
146 | +struct pxa2xx_i2s_s { | |
147 | + target_phys_addr_t base; | |
148 | + qemu_irq irq; | |
149 | + struct pxa2xx_dma_state_s *dma; | |
150 | + void (*data_req)(void *, int, int); | |
151 | + | |
152 | + uint32_t control[2]; | |
153 | + uint32_t status; | |
154 | + uint32_t mask; | |
155 | + uint32_t clk; | |
156 | + | |
157 | + int enable; | |
158 | + int rx_len; | |
159 | + int tx_len; | |
160 | + void (*codec_out)(void *, uint32_t); | |
161 | + uint32_t (*codec_in)(void *); | |
162 | + void *opaque; | |
163 | + | |
164 | + int fifo_len; | |
165 | + uint32_t fifo[16]; | |
166 | +}; | |
167 | + | |
168 | +# define PA_FMT "0x%08lx" | |
169 | +# define REG_FMT "0x%lx" | |
170 | + | |
171 | +struct pxa2xx_state_s *pxa270_init(DisplayState *ds, const char *revision); | |
172 | +struct pxa2xx_state_s *pxa255_init(DisplayState *ds); | |
173 | + | |
174 | +void pxa2xx_reset(int line, int level, void *opaque); | |
175 | + | |
176 | +#endif /* PXA_H */ | ... | ... |
hw/pxa2xx.c
0 → 100644
1 | +/* | |
2 | + * Intel XScale PXA255/270 processor support. | |
3 | + * | |
4 | + * Copyright (c) 2006 Openedhand Ltd. | |
5 | + * Written by Andrzej Zaborowski <balrog@zabor.org> | |
6 | + * | |
7 | + * This code is licenced under the GPL. | |
8 | + */ | |
9 | + | |
10 | +# include "vl.h" | |
11 | + | |
12 | +static struct { | |
13 | + target_phys_addr_t io_base; | |
14 | + int irqn; | |
15 | +} pxa255_serial[] = { | |
16 | + { 0x40100000, PXA2XX_PIC_FFUART }, | |
17 | + { 0x40200000, PXA2XX_PIC_BTUART }, | |
18 | + { 0x40700000, PXA2XX_PIC_STUART }, | |
19 | + { 0x41600000, PXA25X_PIC_HWUART }, | |
20 | + { 0, 0 } | |
21 | +}, pxa270_serial[] = { | |
22 | + { 0x40100000, PXA2XX_PIC_FFUART }, | |
23 | + { 0x40200000, PXA2XX_PIC_BTUART }, | |
24 | + { 0x40700000, PXA2XX_PIC_STUART }, | |
25 | + { 0, 0 } | |
26 | +}; | |
27 | + | |
28 | +static struct { | |
29 | + target_phys_addr_t io_base; | |
30 | + int irqn; | |
31 | +} pxa250_ssp[] = { | |
32 | + { 0x41000000, PXA2XX_PIC_SSP }, | |
33 | + { 0, 0 } | |
34 | +}, pxa255_ssp[] = { | |
35 | + { 0x41000000, PXA2XX_PIC_SSP }, | |
36 | + { 0x41400000, PXA25X_PIC_NSSP }, | |
37 | + { 0, 0 } | |
38 | +}, pxa26x_ssp[] = { | |
39 | + { 0x41000000, PXA2XX_PIC_SSP }, | |
40 | + { 0x41400000, PXA25X_PIC_NSSP }, | |
41 | + { 0x41500000, PXA26X_PIC_ASSP }, | |
42 | + { 0, 0 } | |
43 | +}, pxa27x_ssp[] = { | |
44 | + { 0x41000000, PXA2XX_PIC_SSP }, | |
45 | + { 0x41700000, PXA27X_PIC_SSP2 }, | |
46 | + { 0x41900000, PXA2XX_PIC_SSP3 }, | |
47 | + { 0, 0 } | |
48 | +}; | |
49 | + | |
50 | +#define PMCR 0x00 /* Power Manager Control register */ | |
51 | +#define PSSR 0x04 /* Power Manager Sleep Status register */ | |
52 | +#define PSPR 0x08 /* Power Manager Scratch-Pad register */ | |
53 | +#define PWER 0x0c /* Power Manager Wake-Up Enable register */ | |
54 | +#define PRER 0x10 /* Power Manager Rising-Edge Detect Enable register */ | |
55 | +#define PFER 0x14 /* Power Manager Falling-Edge Detect Enable register */ | |
56 | +#define PEDR 0x18 /* Power Manager Edge-Detect Status register */ | |
57 | +#define PCFR 0x1c /* Power Manager General Configuration register */ | |
58 | +#define PGSR0 0x20 /* Power Manager GPIO Sleep-State register 0 */ | |
59 | +#define PGSR1 0x24 /* Power Manager GPIO Sleep-State register 1 */ | |
60 | +#define PGSR2 0x28 /* Power Manager GPIO Sleep-State register 2 */ | |
61 | +#define PGSR3 0x2c /* Power Manager GPIO Sleep-State register 3 */ | |
62 | +#define RCSR 0x30 /* Reset Controller Status register */ | |
63 | +#define PSLR 0x34 /* Power Manager Sleep Configuration register */ | |
64 | +#define PTSR 0x38 /* Power Manager Standby Configuration register */ | |
65 | +#define PVCR 0x40 /* Power Manager Voltage Change Control register */ | |
66 | +#define PUCR 0x4c /* Power Manager USIM Card Control/Status register */ | |
67 | +#define PKWR 0x50 /* Power Manager Keyboard Wake-Up Enable register */ | |
68 | +#define PKSR 0x54 /* Power Manager Keyboard Level-Detect Status */ | |
69 | +#define PCMD0 0x80 /* Power Manager I2C Command register File 0 */ | |
70 | +#define PCMD31 0xfc /* Power Manager I2C Command register File 31 */ | |
71 | + | |
72 | +static uint32_t pxa2xx_pm_read(void *opaque, target_phys_addr_t addr) | |
73 | +{ | |
74 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
75 | + addr -= s->pm_base; | |
76 | + | |
77 | + switch (addr) { | |
78 | + case PMCR ... PCMD31: | |
79 | + if (addr & 3) | |
80 | + goto fail; | |
81 | + | |
82 | + return s->pm_regs[addr >> 2]; | |
83 | + default: | |
84 | + fail: | |
85 | + printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr); | |
86 | + break; | |
87 | + } | |
88 | + return 0; | |
89 | +} | |
90 | + | |
91 | +static void pxa2xx_pm_write(void *opaque, target_phys_addr_t addr, | |
92 | + uint32_t value) | |
93 | +{ | |
94 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
95 | + addr -= s->pm_base; | |
96 | + | |
97 | + switch (addr) { | |
98 | + case PMCR: | |
99 | + s->pm_regs[addr >> 2] &= 0x15 & ~(value & 0x2a); | |
100 | + s->pm_regs[addr >> 2] |= value & 0x15; | |
101 | + break; | |
102 | + | |
103 | + case PSSR: /* Read-clean registers */ | |
104 | + case RCSR: | |
105 | + case PKSR: | |
106 | + s->pm_regs[addr >> 2] &= ~value; | |
107 | + break; | |
108 | + | |
109 | + default: /* Read-write registers */ | |
110 | + if (addr >= PMCR && addr <= PCMD31 && !(addr & 3)) { | |
111 | + s->pm_regs[addr >> 2] = value; | |
112 | + break; | |
113 | + } | |
114 | + | |
115 | + printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr); | |
116 | + break; | |
117 | + } | |
118 | +} | |
119 | + | |
120 | +static CPUReadMemoryFunc *pxa2xx_pm_readfn[] = { | |
121 | + pxa2xx_pm_read, | |
122 | + pxa2xx_pm_read, | |
123 | + pxa2xx_pm_read, | |
124 | +}; | |
125 | + | |
126 | +static CPUWriteMemoryFunc *pxa2xx_pm_writefn[] = { | |
127 | + pxa2xx_pm_write, | |
128 | + pxa2xx_pm_write, | |
129 | + pxa2xx_pm_write, | |
130 | +}; | |
131 | + | |
132 | +#define CCCR 0x00 /* Core Clock Configuration register */ | |
133 | +#define CKEN 0x04 /* Clock Enable register */ | |
134 | +#define OSCC 0x08 /* Oscillator Configuration register */ | |
135 | +#define CCSR 0x0c /* Core Clock Status register */ | |
136 | + | |
137 | +static uint32_t pxa2xx_cm_read(void *opaque, target_phys_addr_t addr) | |
138 | +{ | |
139 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
140 | + addr -= s->cm_base; | |
141 | + | |
142 | + switch (addr) { | |
143 | + case CCCR: | |
144 | + case CKEN: | |
145 | + case OSCC: | |
146 | + return s->cm_regs[addr >> 2]; | |
147 | + | |
148 | + case CCSR: | |
149 | + return s->cm_regs[CCCR >> 2] | (3 << 28); | |
150 | + | |
151 | + default: | |
152 | + printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr); | |
153 | + break; | |
154 | + } | |
155 | + return 0; | |
156 | +} | |
157 | + | |
158 | +static void pxa2xx_cm_write(void *opaque, target_phys_addr_t addr, | |
159 | + uint32_t value) | |
160 | +{ | |
161 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
162 | + addr -= s->cm_base; | |
163 | + | |
164 | + switch (addr) { | |
165 | + case CCCR: | |
166 | + case CKEN: | |
167 | + s->cm_regs[addr >> 2] = value; | |
168 | + break; | |
169 | + | |
170 | + case OSCC: | |
171 | + s->cm_regs[addr >> 2] &= ~0x6e; | |
172 | + s->cm_regs[addr >> 2] |= value & 0x6e; | |
173 | + break; | |
174 | + | |
175 | + default: | |
176 | + printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr); | |
177 | + break; | |
178 | + } | |
179 | +} | |
180 | + | |
181 | +static CPUReadMemoryFunc *pxa2xx_cm_readfn[] = { | |
182 | + pxa2xx_cm_read, | |
183 | + pxa2xx_cm_read, | |
184 | + pxa2xx_cm_read, | |
185 | +}; | |
186 | + | |
187 | +static CPUWriteMemoryFunc *pxa2xx_cm_writefn[] = { | |
188 | + pxa2xx_cm_write, | |
189 | + pxa2xx_cm_write, | |
190 | + pxa2xx_cm_write, | |
191 | +}; | |
192 | + | |
193 | +static uint32_t pxa2xx_clkpwr_read(void *opaque, int op2, int reg, int crm) | |
194 | +{ | |
195 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
196 | + | |
197 | + switch (reg) { | |
198 | + case 6: /* Clock Configuration register */ | |
199 | + return s->clkcfg; | |
200 | + | |
201 | + case 7: /* Power Mode register */ | |
202 | + return 0; | |
203 | + | |
204 | + default: | |
205 | + printf("%s: Bad register 0x%x\n", __FUNCTION__, reg); | |
206 | + break; | |
207 | + } | |
208 | + return 0; | |
209 | +} | |
210 | + | |
211 | +static void pxa2xx_clkpwr_write(void *opaque, int op2, int reg, int crm, | |
212 | + uint32_t value) | |
213 | +{ | |
214 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
215 | + static const char *pwrmode[8] = { | |
216 | + "Normal", "Idle", "Deep-idle", "Standby", | |
217 | + "Sleep", "reserved (!)", "reserved (!)", "Deep-sleep", | |
218 | + }; | |
219 | + | |
220 | + switch (reg) { | |
221 | + case 6: /* Clock Configuration register */ | |
222 | + s->clkcfg = value & 0xf; | |
223 | + if (value & 2) | |
224 | + printf("%s: CPU frequency change attempt\n", __FUNCTION__); | |
225 | + break; | |
226 | + | |
227 | + case 7: /* Power Mode register */ | |
228 | + if (value & 8) | |
229 | + printf("%s: CPU voltage change attempt\n", __FUNCTION__); | |
230 | + switch (value & 7) { | |
231 | + case 0: | |
232 | + /* Do nothing */ | |
233 | + break; | |
234 | + | |
235 | + case 1: | |
236 | + /* Idle */ | |
237 | + if (!(s->cm_regs[CCCR] & (1 << 31))) { /* CPDIS */ | |
238 | + cpu_interrupt(s->env, CPU_INTERRUPT_HALT); | |
239 | + break; | |
240 | + } | |
241 | + /* Fall through. */ | |
242 | + | |
243 | + case 2: | |
244 | + /* Deep-Idle */ | |
245 | + cpu_interrupt(s->env, CPU_INTERRUPT_HALT); | |
246 | + s->pm_regs[RCSR >> 2] |= 0x8; /* Set GPR */ | |
247 | + goto message; | |
248 | + | |
249 | + case 3: | |
250 | + cpu_reset(s->env); | |
251 | + s->env->cp15.c1_sys = 0; | |
252 | + s->env->cp15.c1_coproc = 0; | |
253 | + s->env->cp15.c2 = 0; | |
254 | + s->env->cp15.c3 = 0; | |
255 | + s->pm_regs[PSSR >> 2] |= 0x8; /* Set STS */ | |
256 | + s->pm_regs[RCSR >> 2] |= 0x8; /* Set GPR */ | |
257 | + | |
258 | + /* | |
259 | + * The scratch-pad register is almost universally used | |
260 | + * for storing the return address on suspend. For the | |
261 | + * lack of a resuming bootloader, perform a jump | |
262 | + * directly to that address. | |
263 | + */ | |
264 | + memset(s->env->regs, 0, 4 * 15); | |
265 | + s->env->regs[15] = s->pm_regs[PSPR >> 2]; | |
266 | + | |
267 | +#if 0 | |
268 | + buffer = 0xe59ff000; /* ldr pc, [pc, #0] */ | |
269 | + cpu_physical_memory_write(0, &buffer, 4); | |
270 | + buffer = s->pm_regs[PSPR >> 2]; | |
271 | + cpu_physical_memory_write(8, &buffer, 4); | |
272 | +#endif | |
273 | + | |
274 | + /* Suspend */ | |
275 | + cpu_interrupt(cpu_single_env, CPU_INTERRUPT_HALT); | |
276 | + | |
277 | + goto message; | |
278 | + | |
279 | + default: | |
280 | + message: | |
281 | + printf("%s: machine entered %s mode\n", __FUNCTION__, | |
282 | + pwrmode[value & 7]); | |
283 | + } | |
284 | + break; | |
285 | + | |
286 | + default: | |
287 | + printf("%s: Bad register 0x%x\n", __FUNCTION__, reg); | |
288 | + break; | |
289 | + } | |
290 | +} | |
291 | + | |
292 | +/* Performace Monitoring Registers */ | |
293 | +#define CPPMNC 0 /* Performance Monitor Control register */ | |
294 | +#define CPCCNT 1 /* Clock Counter register */ | |
295 | +#define CPINTEN 4 /* Interrupt Enable register */ | |
296 | +#define CPFLAG 5 /* Overflow Flag register */ | |
297 | +#define CPEVTSEL 8 /* Event Selection register */ | |
298 | + | |
299 | +#define CPPMN0 0 /* Performance Count register 0 */ | |
300 | +#define CPPMN1 1 /* Performance Count register 1 */ | |
301 | +#define CPPMN2 2 /* Performance Count register 2 */ | |
302 | +#define CPPMN3 3 /* Performance Count register 3 */ | |
303 | + | |
304 | +static uint32_t pxa2xx_perf_read(void *opaque, int op2, int reg, int crm) | |
305 | +{ | |
306 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
307 | + | |
308 | + switch (reg) { | |
309 | + case CPPMNC: | |
310 | + return s->pmnc; | |
311 | + case CPCCNT: | |
312 | + if (s->pmnc & 1) | |
313 | + return qemu_get_clock(vm_clock); | |
314 | + else | |
315 | + return 0; | |
316 | + case CPINTEN: | |
317 | + case CPFLAG: | |
318 | + case CPEVTSEL: | |
319 | + return 0; | |
320 | + | |
321 | + default: | |
322 | + printf("%s: Bad register 0x%x\n", __FUNCTION__, reg); | |
323 | + break; | |
324 | + } | |
325 | + return 0; | |
326 | +} | |
327 | + | |
328 | +static void pxa2xx_perf_write(void *opaque, int op2, int reg, int crm, | |
329 | + uint32_t value) | |
330 | +{ | |
331 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
332 | + | |
333 | + switch (reg) { | |
334 | + case CPPMNC: | |
335 | + s->pmnc = value; | |
336 | + break; | |
337 | + | |
338 | + case CPCCNT: | |
339 | + case CPINTEN: | |
340 | + case CPFLAG: | |
341 | + case CPEVTSEL: | |
342 | + break; | |
343 | + | |
344 | + default: | |
345 | + printf("%s: Bad register 0x%x\n", __FUNCTION__, reg); | |
346 | + break; | |
347 | + } | |
348 | +} | |
349 | + | |
350 | +static uint32_t pxa2xx_cp14_read(void *opaque, int op2, int reg, int crm) | |
351 | +{ | |
352 | + switch (crm) { | |
353 | + case 0: | |
354 | + return pxa2xx_clkpwr_read(opaque, op2, reg, crm); | |
355 | + case 1: | |
356 | + return pxa2xx_perf_read(opaque, op2, reg, crm); | |
357 | + case 2: | |
358 | + switch (reg) { | |
359 | + case CPPMN0: | |
360 | + case CPPMN1: | |
361 | + case CPPMN2: | |
362 | + case CPPMN3: | |
363 | + return 0; | |
364 | + } | |
365 | + /* Fall through */ | |
366 | + default: | |
367 | + printf("%s: Bad register 0x%x\n", __FUNCTION__, reg); | |
368 | + break; | |
369 | + } | |
370 | + return 0; | |
371 | +} | |
372 | + | |
373 | +static void pxa2xx_cp14_write(void *opaque, int op2, int reg, int crm, | |
374 | + uint32_t value) | |
375 | +{ | |
376 | + switch (crm) { | |
377 | + case 0: | |
378 | + pxa2xx_clkpwr_write(opaque, op2, reg, crm, value); | |
379 | + break; | |
380 | + case 1: | |
381 | + pxa2xx_perf_write(opaque, op2, reg, crm, value); | |
382 | + break; | |
383 | + case 2: | |
384 | + switch (reg) { | |
385 | + case CPPMN0: | |
386 | + case CPPMN1: | |
387 | + case CPPMN2: | |
388 | + case CPPMN3: | |
389 | + return; | |
390 | + } | |
391 | + /* Fall through */ | |
392 | + default: | |
393 | + printf("%s: Bad register 0x%x\n", __FUNCTION__, reg); | |
394 | + break; | |
395 | + } | |
396 | +} | |
397 | + | |
398 | +#define MDCNFG 0x00 /* SDRAM Configuration register */ | |
399 | +#define MDREFR 0x04 /* SDRAM Refresh Control register */ | |
400 | +#define MSC0 0x08 /* Static Memory Control register 0 */ | |
401 | +#define MSC1 0x0c /* Static Memory Control register 1 */ | |
402 | +#define MSC2 0x10 /* Static Memory Control register 2 */ | |
403 | +#define MECR 0x14 /* Expansion Memory Bus Config register */ | |
404 | +#define SXCNFG 0x1c /* Synchronous Static Memory Config register */ | |
405 | +#define MCMEM0 0x28 /* PC Card Memory Socket 0 Timing register */ | |
406 | +#define MCMEM1 0x2c /* PC Card Memory Socket 1 Timing register */ | |
407 | +#define MCATT0 0x30 /* PC Card Attribute Socket 0 register */ | |
408 | +#define MCATT1 0x34 /* PC Card Attribute Socket 1 register */ | |
409 | +#define MCIO0 0x38 /* PC Card I/O Socket 0 Timing register */ | |
410 | +#define MCIO1 0x3c /* PC Card I/O Socket 1 Timing register */ | |
411 | +#define MDMRS 0x40 /* SDRAM Mode Register Set Config register */ | |
412 | +#define BOOT_DEF 0x44 /* Boot-time Default Configuration register */ | |
413 | +#define ARB_CNTL 0x48 /* Arbiter Control register */ | |
414 | +#define BSCNTR0 0x4c /* Memory Buffer Strength Control register 0 */ | |
415 | +#define BSCNTR1 0x50 /* Memory Buffer Strength Control register 1 */ | |
416 | +#define LCDBSCNTR 0x54 /* LCD Buffer Strength Control register */ | |
417 | +#define MDMRSLP 0x58 /* Low Power SDRAM Mode Set Config register */ | |
418 | +#define BSCNTR2 0x5c /* Memory Buffer Strength Control register 2 */ | |
419 | +#define BSCNTR3 0x60 /* Memory Buffer Strength Control register 3 */ | |
420 | +#define SA1110 0x64 /* SA-1110 Memory Compatibility register */ | |
421 | + | |
422 | +static uint32_t pxa2xx_mm_read(void *opaque, target_phys_addr_t addr) | |
423 | +{ | |
424 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
425 | + addr -= s->mm_base; | |
426 | + | |
427 | + switch (addr) { | |
428 | + case MDCNFG ... SA1110: | |
429 | + if ((addr & 3) == 0) | |
430 | + return s->mm_regs[addr >> 2]; | |
431 | + | |
432 | + default: | |
433 | + printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr); | |
434 | + break; | |
435 | + } | |
436 | + return 0; | |
437 | +} | |
438 | + | |
439 | +static void pxa2xx_mm_write(void *opaque, target_phys_addr_t addr, | |
440 | + uint32_t value) | |
441 | +{ | |
442 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
443 | + addr -= s->mm_base; | |
444 | + | |
445 | + switch (addr) { | |
446 | + case MDCNFG ... SA1110: | |
447 | + if ((addr & 3) == 0) { | |
448 | + s->mm_regs[addr >> 2] = value; | |
449 | + break; | |
450 | + } | |
451 | + | |
452 | + default: | |
453 | + printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr); | |
454 | + break; | |
455 | + } | |
456 | +} | |
457 | + | |
458 | +static CPUReadMemoryFunc *pxa2xx_mm_readfn[] = { | |
459 | + pxa2xx_mm_read, | |
460 | + pxa2xx_mm_read, | |
461 | + pxa2xx_mm_read, | |
462 | +}; | |
463 | + | |
464 | +static CPUWriteMemoryFunc *pxa2xx_mm_writefn[] = { | |
465 | + pxa2xx_mm_write, | |
466 | + pxa2xx_mm_write, | |
467 | + pxa2xx_mm_write, | |
468 | +}; | |
469 | + | |
470 | +/* Synchronous Serial Ports */ | |
471 | +struct pxa2xx_ssp_s { | |
472 | + target_phys_addr_t base; | |
473 | + qemu_irq irq; | |
474 | + int enable; | |
475 | + | |
476 | + uint32_t sscr[2]; | |
477 | + uint32_t sspsp; | |
478 | + uint32_t ssto; | |
479 | + uint32_t ssitr; | |
480 | + uint32_t sssr; | |
481 | + uint8_t sstsa; | |
482 | + uint8_t ssrsa; | |
483 | + uint8_t ssacd; | |
484 | + | |
485 | + uint32_t rx_fifo[16]; | |
486 | + int rx_level; | |
487 | + int rx_start; | |
488 | + | |
489 | + uint32_t (*readfn)(void *opaque); | |
490 | + void (*writefn)(void *opaque, uint32_t value); | |
491 | + void *opaque; | |
492 | +}; | |
493 | + | |
494 | +#define SSCR0 0x00 /* SSP Control register 0 */ | |
495 | +#define SSCR1 0x04 /* SSP Control register 1 */ | |
496 | +#define SSSR 0x08 /* SSP Status register */ | |
497 | +#define SSITR 0x0c /* SSP Interrupt Test register */ | |
498 | +#define SSDR 0x10 /* SSP Data register */ | |
499 | +#define SSTO 0x28 /* SSP Time-Out register */ | |
500 | +#define SSPSP 0x2c /* SSP Programmable Serial Protocol register */ | |
501 | +#define SSTSA 0x30 /* SSP TX Time Slot Active register */ | |
502 | +#define SSRSA 0x34 /* SSP RX Time Slot Active register */ | |
503 | +#define SSTSS 0x38 /* SSP Time Slot Status register */ | |
504 | +#define SSACD 0x3c /* SSP Audio Clock Divider register */ | |
505 | + | |
506 | +/* Bitfields for above registers */ | |
507 | +#define SSCR0_SPI(x) (((x) & 0x30) == 0x00) | |
508 | +#define SSCR0_SSP(x) (((x) & 0x30) == 0x10) | |
509 | +#define SSCR0_UWIRE(x) (((x) & 0x30) == 0x20) | |
510 | +#define SSCR0_PSP(x) (((x) & 0x30) == 0x30) | |
511 | +#define SSCR0_SSE (1 << 7) | |
512 | +#define SSCR0_RIM (1 << 22) | |
513 | +#define SSCR0_TIM (1 << 23) | |
514 | +#define SSCR0_MOD (1 << 31) | |
515 | +#define SSCR0_DSS(x) (((((x) >> 16) & 0x10) | ((x) & 0xf)) + 1) | |
516 | +#define SSCR1_RIE (1 << 0) | |
517 | +#define SSCR1_TIE (1 << 1) | |
518 | +#define SSCR1_LBM (1 << 2) | |
519 | +#define SSCR1_MWDS (1 << 5) | |
520 | +#define SSCR1_TFT(x) ((((x) >> 6) & 0xf) + 1) | |
521 | +#define SSCR1_RFT(x) ((((x) >> 10) & 0xf) + 1) | |
522 | +#define SSCR1_EFWR (1 << 14) | |
523 | +#define SSCR1_PINTE (1 << 18) | |
524 | +#define SSCR1_TINTE (1 << 19) | |
525 | +#define SSCR1_RSRE (1 << 20) | |
526 | +#define SSCR1_TSRE (1 << 21) | |
527 | +#define SSCR1_EBCEI (1 << 29) | |
528 | +#define SSITR_INT (7 << 5) | |
529 | +#define SSSR_TNF (1 << 2) | |
530 | +#define SSSR_RNE (1 << 3) | |
531 | +#define SSSR_TFS (1 << 5) | |
532 | +#define SSSR_RFS (1 << 6) | |
533 | +#define SSSR_ROR (1 << 7) | |
534 | +#define SSSR_PINT (1 << 18) | |
535 | +#define SSSR_TINT (1 << 19) | |
536 | +#define SSSR_EOC (1 << 20) | |
537 | +#define SSSR_TUR (1 << 21) | |
538 | +#define SSSR_BCE (1 << 23) | |
539 | +#define SSSR_RW 0x00bc0080 | |
540 | + | |
541 | +static void pxa2xx_ssp_int_update(struct pxa2xx_ssp_s *s) | |
542 | +{ | |
543 | + int level = 0; | |
544 | + | |
545 | + level |= s->ssitr & SSITR_INT; | |
546 | + level |= (s->sssr & SSSR_BCE) && (s->sscr[1] & SSCR1_EBCEI); | |
547 | + level |= (s->sssr & SSSR_TUR) && !(s->sscr[0] & SSCR0_TIM); | |
548 | + level |= (s->sssr & SSSR_EOC) && (s->sssr & (SSSR_TINT | SSSR_PINT)); | |
549 | + level |= (s->sssr & SSSR_TINT) && (s->sscr[1] & SSCR1_TINTE); | |
550 | + level |= (s->sssr & SSSR_PINT) && (s->sscr[1] & SSCR1_PINTE); | |
551 | + level |= (s->sssr & SSSR_ROR) && !(s->sscr[0] & SSCR0_RIM); | |
552 | + level |= (s->sssr & SSSR_RFS) && (s->sscr[1] & SSCR1_RIE); | |
553 | + level |= (s->sssr & SSSR_TFS) && (s->sscr[1] & SSCR1_TIE); | |
554 | + qemu_set_irq(s->irq, !!level); | |
555 | +} | |
556 | + | |
557 | +static void pxa2xx_ssp_fifo_update(struct pxa2xx_ssp_s *s) | |
558 | +{ | |
559 | + s->sssr &= ~(0xf << 12); /* Clear RFL */ | |
560 | + s->sssr &= ~(0xf << 8); /* Clear TFL */ | |
561 | + s->sssr &= ~SSSR_TNF; | |
562 | + if (s->enable) { | |
563 | + s->sssr |= ((s->rx_level - 1) & 0xf) << 12; | |
564 | + if (s->rx_level >= SSCR1_RFT(s->sscr[1])) | |
565 | + s->sssr |= SSSR_RFS; | |
566 | + else | |
567 | + s->sssr &= ~SSSR_RFS; | |
568 | + if (0 <= SSCR1_TFT(s->sscr[1])) | |
569 | + s->sssr |= SSSR_TFS; | |
570 | + else | |
571 | + s->sssr &= ~SSSR_TFS; | |
572 | + if (s->rx_level) | |
573 | + s->sssr |= SSSR_RNE; | |
574 | + else | |
575 | + s->sssr &= ~SSSR_RNE; | |
576 | + s->sssr |= SSSR_TNF; | |
577 | + } | |
578 | + | |
579 | + pxa2xx_ssp_int_update(s); | |
580 | +} | |
581 | + | |
582 | +static uint32_t pxa2xx_ssp_read(void *opaque, target_phys_addr_t addr) | |
583 | +{ | |
584 | + struct pxa2xx_ssp_s *s = (struct pxa2xx_ssp_s *) opaque; | |
585 | + uint32_t retval; | |
586 | + addr -= s->base; | |
587 | + | |
588 | + switch (addr) { | |
589 | + case SSCR0: | |
590 | + return s->sscr[0]; | |
591 | + case SSCR1: | |
592 | + return s->sscr[1]; | |
593 | + case SSPSP: | |
594 | + return s->sspsp; | |
595 | + case SSTO: | |
596 | + return s->ssto; | |
597 | + case SSITR: | |
598 | + return s->ssitr; | |
599 | + case SSSR: | |
600 | + return s->sssr | s->ssitr; | |
601 | + case SSDR: | |
602 | + if (!s->enable) | |
603 | + return 0xffffffff; | |
604 | + if (s->rx_level < 1) { | |
605 | + printf("%s: SSP Rx Underrun\n", __FUNCTION__); | |
606 | + return 0xffffffff; | |
607 | + } | |
608 | + s->rx_level --; | |
609 | + retval = s->rx_fifo[s->rx_start ++]; | |
610 | + s->rx_start &= 0xf; | |
611 | + pxa2xx_ssp_fifo_update(s); | |
612 | + return retval; | |
613 | + case SSTSA: | |
614 | + return s->sstsa; | |
615 | + case SSRSA: | |
616 | + return s->ssrsa; | |
617 | + case SSTSS: | |
618 | + return 0; | |
619 | + case SSACD: | |
620 | + return s->ssacd; | |
621 | + default: | |
622 | + printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr); | |
623 | + break; | |
624 | + } | |
625 | + return 0; | |
626 | +} | |
627 | + | |
628 | +static void pxa2xx_ssp_write(void *opaque, target_phys_addr_t addr, | |
629 | + uint32_t value) | |
630 | +{ | |
631 | + struct pxa2xx_ssp_s *s = (struct pxa2xx_ssp_s *) opaque; | |
632 | + addr -= s->base; | |
633 | + | |
634 | + switch (addr) { | |
635 | + case SSCR0: | |
636 | + s->sscr[0] = value & 0xc7ffffff; | |
637 | + s->enable = value & SSCR0_SSE; | |
638 | + if (value & SSCR0_MOD) | |
639 | + printf("%s: Attempt to use network mode\n", __FUNCTION__); | |
640 | + if (s->enable && SSCR0_DSS(value) < 4) | |
641 | + printf("%s: Wrong data size: %i bits\n", __FUNCTION__, | |
642 | + SSCR0_DSS(value)); | |
643 | + if (!(value & SSCR0_SSE)) { | |
644 | + s->sssr = 0; | |
645 | + s->ssitr = 0; | |
646 | + s->rx_level = 0; | |
647 | + } | |
648 | + pxa2xx_ssp_fifo_update(s); | |
649 | + break; | |
650 | + | |
651 | + case SSCR1: | |
652 | + s->sscr[1] = value; | |
653 | + if (value & (SSCR1_LBM | SSCR1_EFWR)) | |
654 | + printf("%s: Attempt to use SSP test mode\n", __FUNCTION__); | |
655 | + pxa2xx_ssp_fifo_update(s); | |
656 | + break; | |
657 | + | |
658 | + case SSPSP: | |
659 | + s->sspsp = value; | |
660 | + break; | |
661 | + | |
662 | + case SSTO: | |
663 | + s->ssto = value; | |
664 | + break; | |
665 | + | |
666 | + case SSITR: | |
667 | + s->ssitr = value & SSITR_INT; | |
668 | + pxa2xx_ssp_int_update(s); | |
669 | + break; | |
670 | + | |
671 | + case SSSR: | |
672 | + s->sssr &= ~(value & SSSR_RW); | |
673 | + pxa2xx_ssp_int_update(s); | |
674 | + break; | |
675 | + | |
676 | + case SSDR: | |
677 | + if (SSCR0_UWIRE(s->sscr[0])) { | |
678 | + if (s->sscr[1] & SSCR1_MWDS) | |
679 | + value &= 0xffff; | |
680 | + else | |
681 | + value &= 0xff; | |
682 | + } else | |
683 | + /* Note how 32bits overflow does no harm here */ | |
684 | + value &= (1 << SSCR0_DSS(s->sscr[0])) - 1; | |
685 | + | |
686 | + /* Data goes from here to the Tx FIFO and is shifted out from | |
687 | + * there directly to the slave, no need to buffer it. | |
688 | + */ | |
689 | + if (s->enable) { | |
690 | + if (s->writefn) | |
691 | + s->writefn(s->opaque, value); | |
692 | + | |
693 | + if (s->rx_level < 0x10) { | |
694 | + if (s->readfn) | |
695 | + s->rx_fifo[(s->rx_start + s->rx_level ++) & 0xf] = | |
696 | + s->readfn(s->opaque); | |
697 | + else | |
698 | + s->rx_fifo[(s->rx_start + s->rx_level ++) & 0xf] = 0x0; | |
699 | + } else | |
700 | + s->sssr |= SSSR_ROR; | |
701 | + } | |
702 | + pxa2xx_ssp_fifo_update(s); | |
703 | + break; | |
704 | + | |
705 | + case SSTSA: | |
706 | + s->sstsa = value; | |
707 | + break; | |
708 | + | |
709 | + case SSRSA: | |
710 | + s->ssrsa = value; | |
711 | + break; | |
712 | + | |
713 | + case SSACD: | |
714 | + s->ssacd = value; | |
715 | + break; | |
716 | + | |
717 | + default: | |
718 | + printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr); | |
719 | + break; | |
720 | + } | |
721 | +} | |
722 | + | |
723 | +void pxa2xx_ssp_attach(struct pxa2xx_ssp_s *port, | |
724 | + uint32_t (*readfn)(void *opaque), | |
725 | + void (*writefn)(void *opaque, uint32_t value), void *opaque) | |
726 | +{ | |
727 | + if (!port) { | |
728 | + printf("%s: no such SSP\n", __FUNCTION__); | |
729 | + exit(-1); | |
730 | + } | |
731 | + | |
732 | + port->opaque = opaque; | |
733 | + port->readfn = readfn; | |
734 | + port->writefn = writefn; | |
735 | +} | |
736 | + | |
737 | +static CPUReadMemoryFunc *pxa2xx_ssp_readfn[] = { | |
738 | + pxa2xx_ssp_read, | |
739 | + pxa2xx_ssp_read, | |
740 | + pxa2xx_ssp_read, | |
741 | +}; | |
742 | + | |
743 | +static CPUWriteMemoryFunc *pxa2xx_ssp_writefn[] = { | |
744 | + pxa2xx_ssp_write, | |
745 | + pxa2xx_ssp_write, | |
746 | + pxa2xx_ssp_write, | |
747 | +}; | |
748 | + | |
749 | +/* Real-Time Clock */ | |
750 | +#define RCNR 0x00 /* RTC Counter register */ | |
751 | +#define RTAR 0x04 /* RTC Alarm register */ | |
752 | +#define RTSR 0x08 /* RTC Status register */ | |
753 | +#define RTTR 0x0c /* RTC Timer Trim register */ | |
754 | +#define RDCR 0x10 /* RTC Day Counter register */ | |
755 | +#define RYCR 0x14 /* RTC Year Counter register */ | |
756 | +#define RDAR1 0x18 /* RTC Wristwatch Day Alarm register 1 */ | |
757 | +#define RYAR1 0x1c /* RTC Wristwatch Year Alarm register 1 */ | |
758 | +#define RDAR2 0x20 /* RTC Wristwatch Day Alarm register 2 */ | |
759 | +#define RYAR2 0x24 /* RTC Wristwatch Year Alarm register 2 */ | |
760 | +#define SWCR 0x28 /* RTC Stopwatch Counter register */ | |
761 | +#define SWAR1 0x2c /* RTC Stopwatch Alarm register 1 */ | |
762 | +#define SWAR2 0x30 /* RTC Stopwatch Alarm register 2 */ | |
763 | +#define RTCPICR 0x34 /* RTC Periodic Interrupt Counter register */ | |
764 | +#define PIAR 0x38 /* RTC Periodic Interrupt Alarm register */ | |
765 | + | |
766 | +static inline void pxa2xx_rtc_int_update(struct pxa2xx_state_s *s) | |
767 | +{ | |
768 | + qemu_set_irq(s->pic[PXA2XX_PIC_RTCALARM], !!(s->rtsr & 0x2553)); | |
769 | +} | |
770 | + | |
771 | +static void pxa2xx_rtc_hzupdate(struct pxa2xx_state_s *s) | |
772 | +{ | |
773 | + int64_t rt = qemu_get_clock(rt_clock); | |
774 | + s->last_rcnr += ((rt - s->last_hz) << 15) / | |
775 | + (1000 * ((s->rttr & 0xffff) + 1)); | |
776 | + s->last_rdcr += ((rt - s->last_hz) << 15) / | |
777 | + (1000 * ((s->rttr & 0xffff) + 1)); | |
778 | + s->last_hz = rt; | |
779 | +} | |
780 | + | |
781 | +static void pxa2xx_rtc_swupdate(struct pxa2xx_state_s *s) | |
782 | +{ | |
783 | + int64_t rt = qemu_get_clock(rt_clock); | |
784 | + if (s->rtsr & (1 << 12)) | |
785 | + s->last_swcr += (rt - s->last_sw) / 10; | |
786 | + s->last_sw = rt; | |
787 | +} | |
788 | + | |
789 | +static void pxa2xx_rtc_piupdate(struct pxa2xx_state_s *s) | |
790 | +{ | |
791 | + int64_t rt = qemu_get_clock(rt_clock); | |
792 | + if (s->rtsr & (1 << 15)) | |
793 | + s->last_swcr += rt - s->last_pi; | |
794 | + s->last_pi = rt; | |
795 | +} | |
796 | + | |
797 | +static inline void pxa2xx_rtc_alarm_update(struct pxa2xx_state_s *s, | |
798 | + uint32_t rtsr) | |
799 | +{ | |
800 | + if ((rtsr & (1 << 2)) && !(rtsr & (1 << 0))) | |
801 | + qemu_mod_timer(s->rtc_hz, s->last_hz + | |
802 | + (((s->rtar - s->last_rcnr) * 1000 * | |
803 | + ((s->rttr & 0xffff) + 1)) >> 15)); | |
804 | + else | |
805 | + qemu_del_timer(s->rtc_hz); | |
806 | + | |
807 | + if ((rtsr & (1 << 5)) && !(rtsr & (1 << 4))) | |
808 | + qemu_mod_timer(s->rtc_rdal1, s->last_hz + | |
809 | + (((s->rdar1 - s->last_rdcr) * 1000 * | |
810 | + ((s->rttr & 0xffff) + 1)) >> 15)); /* TODO: fixup */ | |
811 | + else | |
812 | + qemu_del_timer(s->rtc_rdal1); | |
813 | + | |
814 | + if ((rtsr & (1 << 7)) && !(rtsr & (1 << 6))) | |
815 | + qemu_mod_timer(s->rtc_rdal2, s->last_hz + | |
816 | + (((s->rdar2 - s->last_rdcr) * 1000 * | |
817 | + ((s->rttr & 0xffff) + 1)) >> 15)); /* TODO: fixup */ | |
818 | + else | |
819 | + qemu_del_timer(s->rtc_rdal2); | |
820 | + | |
821 | + if ((rtsr & 0x1200) == 0x1200 && !(rtsr & (1 << 8))) | |
822 | + qemu_mod_timer(s->rtc_swal1, s->last_sw + | |
823 | + (s->swar1 - s->last_swcr) * 10); /* TODO: fixup */ | |
824 | + else | |
825 | + qemu_del_timer(s->rtc_swal1); | |
826 | + | |
827 | + if ((rtsr & 0x1800) == 0x1800 && !(rtsr & (1 << 10))) | |
828 | + qemu_mod_timer(s->rtc_swal2, s->last_sw + | |
829 | + (s->swar2 - s->last_swcr) * 10); /* TODO: fixup */ | |
830 | + else | |
831 | + qemu_del_timer(s->rtc_swal2); | |
832 | + | |
833 | + if ((rtsr & 0xc000) == 0xc000 && !(rtsr & (1 << 13))) | |
834 | + qemu_mod_timer(s->rtc_pi, s->last_pi + | |
835 | + (s->piar & 0xffff) - s->last_rtcpicr); | |
836 | + else | |
837 | + qemu_del_timer(s->rtc_pi); | |
838 | +} | |
839 | + | |
840 | +static inline void pxa2xx_rtc_hz_tick(void *opaque) | |
841 | +{ | |
842 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
843 | + s->rtsr |= (1 << 0); | |
844 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
845 | + pxa2xx_rtc_int_update(s); | |
846 | +} | |
847 | + | |
848 | +static inline void pxa2xx_rtc_rdal1_tick(void *opaque) | |
849 | +{ | |
850 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
851 | + s->rtsr |= (1 << 4); | |
852 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
853 | + pxa2xx_rtc_int_update(s); | |
854 | +} | |
855 | + | |
856 | +static inline void pxa2xx_rtc_rdal2_tick(void *opaque) | |
857 | +{ | |
858 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
859 | + s->rtsr |= (1 << 6); | |
860 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
861 | + pxa2xx_rtc_int_update(s); | |
862 | +} | |
863 | + | |
864 | +static inline void pxa2xx_rtc_swal1_tick(void *opaque) | |
865 | +{ | |
866 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
867 | + s->rtsr |= (1 << 8); | |
868 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
869 | + pxa2xx_rtc_int_update(s); | |
870 | +} | |
871 | + | |
872 | +static inline void pxa2xx_rtc_swal2_tick(void *opaque) | |
873 | +{ | |
874 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
875 | + s->rtsr |= (1 << 10); | |
876 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
877 | + pxa2xx_rtc_int_update(s); | |
878 | +} | |
879 | + | |
880 | +static inline void pxa2xx_rtc_pi_tick(void *opaque) | |
881 | +{ | |
882 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
883 | + s->rtsr |= (1 << 13); | |
884 | + pxa2xx_rtc_piupdate(s); | |
885 | + s->last_rtcpicr = 0; | |
886 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
887 | + pxa2xx_rtc_int_update(s); | |
888 | +} | |
889 | + | |
890 | +static uint32_t pxa2xx_rtc_read(void *opaque, target_phys_addr_t addr) | |
891 | +{ | |
892 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
893 | + addr -= s->rtc_base; | |
894 | + | |
895 | + switch (addr) { | |
896 | + case RTTR: | |
897 | + return s->rttr; | |
898 | + case RTSR: | |
899 | + return s->rtsr; | |
900 | + case RTAR: | |
901 | + return s->rtar; | |
902 | + case RDAR1: | |
903 | + return s->rdar1; | |
904 | + case RDAR2: | |
905 | + return s->rdar2; | |
906 | + case RYAR1: | |
907 | + return s->ryar1; | |
908 | + case RYAR2: | |
909 | + return s->ryar2; | |
910 | + case SWAR1: | |
911 | + return s->swar1; | |
912 | + case SWAR2: | |
913 | + return s->swar2; | |
914 | + case PIAR: | |
915 | + return s->piar; | |
916 | + case RCNR: | |
917 | + return s->last_rcnr + ((qemu_get_clock(rt_clock) - s->last_hz) << 15) / | |
918 | + (1000 * ((s->rttr & 0xffff) + 1)); | |
919 | + case RDCR: | |
920 | + return s->last_rdcr + ((qemu_get_clock(rt_clock) - s->last_hz) << 15) / | |
921 | + (1000 * ((s->rttr & 0xffff) + 1)); | |
922 | + case RYCR: | |
923 | + return s->last_rycr; | |
924 | + case SWCR: | |
925 | + if (s->rtsr & (1 << 12)) | |
926 | + return s->last_swcr + (qemu_get_clock(rt_clock) - s->last_sw) / 10; | |
927 | + else | |
928 | + return s->last_swcr; | |
929 | + default: | |
930 | + printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr); | |
931 | + break; | |
932 | + } | |
933 | + return 0; | |
934 | +} | |
935 | + | |
936 | +static void pxa2xx_rtc_write(void *opaque, target_phys_addr_t addr, | |
937 | + uint32_t value) | |
938 | +{ | |
939 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
940 | + addr -= s->rtc_base; | |
941 | + | |
942 | + switch (addr) { | |
943 | + case RTTR: | |
944 | + if (!(s->rttr & (1 << 31))) { | |
945 | + pxa2xx_rtc_hzupdate(s); | |
946 | + s->rttr = value; | |
947 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
948 | + } | |
949 | + break; | |
950 | + | |
951 | + case RTSR: | |
952 | + if ((s->rtsr ^ value) & (1 << 15)) | |
953 | + pxa2xx_rtc_piupdate(s); | |
954 | + | |
955 | + if ((s->rtsr ^ value) & (1 << 12)) | |
956 | + pxa2xx_rtc_swupdate(s); | |
957 | + | |
958 | + if (((s->rtsr ^ value) & 0x4aac) | (value & ~0xdaac)) | |
959 | + pxa2xx_rtc_alarm_update(s, value); | |
960 | + | |
961 | + s->rtsr = (value & 0xdaac) | (s->rtsr & ~(value & ~0xdaac)); | |
962 | + pxa2xx_rtc_int_update(s); | |
963 | + break; | |
964 | + | |
965 | + case RTAR: | |
966 | + s->rtar = value; | |
967 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
968 | + break; | |
969 | + | |
970 | + case RDAR1: | |
971 | + s->rdar1 = value; | |
972 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
973 | + break; | |
974 | + | |
975 | + case RDAR2: | |
976 | + s->rdar2 = value; | |
977 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
978 | + break; | |
979 | + | |
980 | + case RYAR1: | |
981 | + s->ryar1 = value; | |
982 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
983 | + break; | |
984 | + | |
985 | + case RYAR2: | |
986 | + s->ryar2 = value; | |
987 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
988 | + break; | |
989 | + | |
990 | + case SWAR1: | |
991 | + pxa2xx_rtc_swupdate(s); | |
992 | + s->swar1 = value; | |
993 | + s->last_swcr = 0; | |
994 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
995 | + break; | |
996 | + | |
997 | + case SWAR2: | |
998 | + s->swar2 = value; | |
999 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
1000 | + break; | |
1001 | + | |
1002 | + case PIAR: | |
1003 | + s->piar = value; | |
1004 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
1005 | + break; | |
1006 | + | |
1007 | + case RCNR: | |
1008 | + pxa2xx_rtc_hzupdate(s); | |
1009 | + s->last_rcnr = value; | |
1010 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
1011 | + break; | |
1012 | + | |
1013 | + case RDCR: | |
1014 | + pxa2xx_rtc_hzupdate(s); | |
1015 | + s->last_rdcr = value; | |
1016 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
1017 | + break; | |
1018 | + | |
1019 | + case RYCR: | |
1020 | + s->last_rycr = value; | |
1021 | + break; | |
1022 | + | |
1023 | + case SWCR: | |
1024 | + pxa2xx_rtc_swupdate(s); | |
1025 | + s->last_swcr = value; | |
1026 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
1027 | + break; | |
1028 | + | |
1029 | + case RTCPICR: | |
1030 | + pxa2xx_rtc_piupdate(s); | |
1031 | + s->last_rtcpicr = value & 0xffff; | |
1032 | + pxa2xx_rtc_alarm_update(s, s->rtsr); | |
1033 | + break; | |
1034 | + | |
1035 | + default: | |
1036 | + printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr); | |
1037 | + } | |
1038 | +} | |
1039 | + | |
1040 | +static void pxa2xx_rtc_reset(struct pxa2xx_state_s *s) | |
1041 | +{ | |
1042 | + struct tm *tm; | |
1043 | + time_t ti; | |
1044 | + int wom; | |
1045 | + | |
1046 | + s->rttr = 0x7fff; | |
1047 | + s->rtsr = 0; | |
1048 | + | |
1049 | + time(&ti); | |
1050 | + if (rtc_utc) | |
1051 | + tm = gmtime(&ti); | |
1052 | + else | |
1053 | + tm = localtime(&ti); | |
1054 | + wom = ((tm->tm_mday - 1) / 7) + 1; | |
1055 | + | |
1056 | + s->last_rcnr = (uint32_t) ti; | |
1057 | + s->last_rdcr = (wom << 20) | ((tm->tm_wday + 1) << 17) | | |
1058 | + (tm->tm_hour << 12) | (tm->tm_min << 6) | tm->tm_sec; | |
1059 | + s->last_rycr = ((tm->tm_year + 1900) << 9) | | |
1060 | + ((tm->tm_mon + 1) << 5) | tm->tm_mday; | |
1061 | + s->last_swcr = (tm->tm_hour << 19) | | |
1062 | + (tm->tm_min << 13) | (tm->tm_sec << 7); | |
1063 | + s->last_rtcpicr = 0; | |
1064 | + s->last_hz = s->last_sw = s->last_pi = qemu_get_clock(rt_clock); | |
1065 | + | |
1066 | + s->rtc_hz = qemu_new_timer(rt_clock, pxa2xx_rtc_hz_tick, s); | |
1067 | + s->rtc_rdal1 = qemu_new_timer(rt_clock, pxa2xx_rtc_rdal1_tick, s); | |
1068 | + s->rtc_rdal2 = qemu_new_timer(rt_clock, pxa2xx_rtc_rdal2_tick, s); | |
1069 | + s->rtc_swal1 = qemu_new_timer(rt_clock, pxa2xx_rtc_swal1_tick, s); | |
1070 | + s->rtc_swal2 = qemu_new_timer(rt_clock, pxa2xx_rtc_swal2_tick, s); | |
1071 | + s->rtc_pi = qemu_new_timer(rt_clock, pxa2xx_rtc_pi_tick, s); | |
1072 | +} | |
1073 | + | |
1074 | +static CPUReadMemoryFunc *pxa2xx_rtc_readfn[] = { | |
1075 | + pxa2xx_rtc_read, | |
1076 | + pxa2xx_rtc_read, | |
1077 | + pxa2xx_rtc_read, | |
1078 | +}; | |
1079 | + | |
1080 | +static CPUWriteMemoryFunc *pxa2xx_rtc_writefn[] = { | |
1081 | + pxa2xx_rtc_write, | |
1082 | + pxa2xx_rtc_write, | |
1083 | + pxa2xx_rtc_write, | |
1084 | +}; | |
1085 | + | |
1086 | +/* PXA Inter-IC Sound Controller */ | |
1087 | +static void pxa2xx_i2s_reset(struct pxa2xx_i2s_s *i2s) | |
1088 | +{ | |
1089 | + i2s->rx_len = 0; | |
1090 | + i2s->tx_len = 0; | |
1091 | + i2s->fifo_len = 0; | |
1092 | + i2s->clk = 0x1a; | |
1093 | + i2s->control[0] = 0x00; | |
1094 | + i2s->control[1] = 0x00; | |
1095 | + i2s->status = 0x00; | |
1096 | + i2s->mask = 0x00; | |
1097 | +} | |
1098 | + | |
1099 | +#define SACR_TFTH(val) ((val >> 8) & 0xf) | |
1100 | +#define SACR_RFTH(val) ((val >> 12) & 0xf) | |
1101 | +#define SACR_DREC(val) (val & (1 << 3)) | |
1102 | +#define SACR_DPRL(val) (val & (1 << 4)) | |
1103 | + | |
1104 | +static inline void pxa2xx_i2s_update(struct pxa2xx_i2s_s *i2s) | |
1105 | +{ | |
1106 | + int rfs, tfs; | |
1107 | + rfs = SACR_RFTH(i2s->control[0]) < i2s->rx_len && | |
1108 | + !SACR_DREC(i2s->control[1]); | |
1109 | + tfs = (i2s->tx_len || i2s->fifo_len < SACR_TFTH(i2s->control[0])) && | |
1110 | + i2s->enable && !SACR_DPRL(i2s->control[1]); | |
1111 | + | |
1112 | + pxa2xx_dma_request(i2s->dma, PXA2XX_RX_RQ_I2S, rfs); | |
1113 | + pxa2xx_dma_request(i2s->dma, PXA2XX_TX_RQ_I2S, tfs); | |
1114 | + | |
1115 | + i2s->status &= 0xe0; | |
1116 | + if (i2s->rx_len) | |
1117 | + i2s->status |= 1 << 1; /* RNE */ | |
1118 | + if (i2s->enable) | |
1119 | + i2s->status |= 1 << 2; /* BSY */ | |
1120 | + if (tfs) | |
1121 | + i2s->status |= 1 << 3; /* TFS */ | |
1122 | + if (rfs) | |
1123 | + i2s->status |= 1 << 4; /* RFS */ | |
1124 | + if (!(i2s->tx_len && i2s->enable)) | |
1125 | + i2s->status |= i2s->fifo_len << 8; /* TFL */ | |
1126 | + i2s->status |= MAX(i2s->rx_len, 0xf) << 12; /* RFL */ | |
1127 | + | |
1128 | + qemu_set_irq(i2s->irq, i2s->status & i2s->mask); | |
1129 | +} | |
1130 | + | |
1131 | +#define SACR0 0x00 /* Serial Audio Global Control register */ | |
1132 | +#define SACR1 0x04 /* Serial Audio I2S/MSB-Justified Control register */ | |
1133 | +#define SASR0 0x0c /* Serial Audio Interface and FIFO Status register */ | |
1134 | +#define SAIMR 0x14 /* Serial Audio Interrupt Mask register */ | |
1135 | +#define SAICR 0x18 /* Serial Audio Interrupt Clear register */ | |
1136 | +#define SADIV 0x60 /* Serial Audio Clock Divider register */ | |
1137 | +#define SADR 0x80 /* Serial Audio Data register */ | |
1138 | + | |
1139 | +static uint32_t pxa2xx_i2s_read(void *opaque, target_phys_addr_t addr) | |
1140 | +{ | |
1141 | + struct pxa2xx_i2s_s *s = (struct pxa2xx_i2s_s *) opaque; | |
1142 | + addr -= s->base; | |
1143 | + | |
1144 | + switch (addr) { | |
1145 | + case SACR0: | |
1146 | + return s->control[0]; | |
1147 | + case SACR1: | |
1148 | + return s->control[1]; | |
1149 | + case SASR0: | |
1150 | + return s->status; | |
1151 | + case SAIMR: | |
1152 | + return s->mask; | |
1153 | + case SAICR: | |
1154 | + return 0; | |
1155 | + case SADIV: | |
1156 | + return s->clk; | |
1157 | + case SADR: | |
1158 | + if (s->rx_len > 0) { | |
1159 | + s->rx_len --; | |
1160 | + pxa2xx_i2s_update(s); | |
1161 | + return s->codec_in(s->opaque); | |
1162 | + } | |
1163 | + return 0; | |
1164 | + default: | |
1165 | + printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr); | |
1166 | + break; | |
1167 | + } | |
1168 | + return 0; | |
1169 | +} | |
1170 | + | |
1171 | +static void pxa2xx_i2s_write(void *opaque, target_phys_addr_t addr, | |
1172 | + uint32_t value) | |
1173 | +{ | |
1174 | + struct pxa2xx_i2s_s *s = (struct pxa2xx_i2s_s *) opaque; | |
1175 | + uint32_t *sample; | |
1176 | + addr -= s->base; | |
1177 | + | |
1178 | + switch (addr) { | |
1179 | + case SACR0: | |
1180 | + if (value & (1 << 3)) /* RST */ | |
1181 | + pxa2xx_i2s_reset(s); | |
1182 | + s->control[0] = value & 0xff3d; | |
1183 | + if (!s->enable && (value & 1) && s->tx_len) { /* ENB */ | |
1184 | + for (sample = s->fifo; s->fifo_len > 0; s->fifo_len --, sample ++) | |
1185 | + s->codec_out(s->opaque, *sample); | |
1186 | + s->status &= ~(1 << 7); /* I2SOFF */ | |
1187 | + } | |
1188 | + if (value & (1 << 4)) /* EFWR */ | |
1189 | + printf("%s: Attempt to use special function\n", __FUNCTION__); | |
1190 | + s->enable = ((value ^ 4) & 5) == 5; /* ENB && !RST*/ | |
1191 | + pxa2xx_i2s_update(s); | |
1192 | + break; | |
1193 | + case SACR1: | |
1194 | + s->control[1] = value & 0x0039; | |
1195 | + if (value & (1 << 5)) /* ENLBF */ | |
1196 | + printf("%s: Attempt to use loopback function\n", __FUNCTION__); | |
1197 | + if (value & (1 << 4)) /* DPRL */ | |
1198 | + s->fifo_len = 0; | |
1199 | + pxa2xx_i2s_update(s); | |
1200 | + break; | |
1201 | + case SAIMR: | |
1202 | + s->mask = value & 0x0078; | |
1203 | + pxa2xx_i2s_update(s); | |
1204 | + break; | |
1205 | + case SAICR: | |
1206 | + s->status &= ~(value & (3 << 5)); | |
1207 | + pxa2xx_i2s_update(s); | |
1208 | + break; | |
1209 | + case SADIV: | |
1210 | + s->clk = value & 0x007f; | |
1211 | + break; | |
1212 | + case SADR: | |
1213 | + if (s->tx_len && s->enable) { | |
1214 | + s->tx_len --; | |
1215 | + pxa2xx_i2s_update(s); | |
1216 | + s->codec_out(s->opaque, value); | |
1217 | + } else if (s->fifo_len < 16) { | |
1218 | + s->fifo[s->fifo_len ++] = value; | |
1219 | + pxa2xx_i2s_update(s); | |
1220 | + } | |
1221 | + break; | |
1222 | + default: | |
1223 | + printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr); | |
1224 | + } | |
1225 | +} | |
1226 | + | |
1227 | +static CPUReadMemoryFunc *pxa2xx_i2s_readfn[] = { | |
1228 | + pxa2xx_i2s_read, | |
1229 | + pxa2xx_i2s_read, | |
1230 | + pxa2xx_i2s_read, | |
1231 | +}; | |
1232 | + | |
1233 | +static CPUWriteMemoryFunc *pxa2xx_i2s_writefn[] = { | |
1234 | + pxa2xx_i2s_write, | |
1235 | + pxa2xx_i2s_write, | |
1236 | + pxa2xx_i2s_write, | |
1237 | +}; | |
1238 | + | |
1239 | +static void pxa2xx_i2s_data_req(void *opaque, int tx, int rx) | |
1240 | +{ | |
1241 | + struct pxa2xx_i2s_s *s = (struct pxa2xx_i2s_s *) opaque; | |
1242 | + uint32_t *sample; | |
1243 | + | |
1244 | + /* Signal FIFO errors */ | |
1245 | + if (s->enable && s->tx_len) | |
1246 | + s->status |= 1 << 5; /* TUR */ | |
1247 | + if (s->enable && s->rx_len) | |
1248 | + s->status |= 1 << 6; /* ROR */ | |
1249 | + | |
1250 | + /* Should be tx - MIN(tx, s->fifo_len) but we don't really need to | |
1251 | + * handle the cases where it makes a difference. */ | |
1252 | + s->tx_len = tx - s->fifo_len; | |
1253 | + s->rx_len = rx; | |
1254 | + /* Note that is s->codec_out wasn't set, we wouldn't get called. */ | |
1255 | + if (s->enable) | |
1256 | + for (sample = s->fifo; s->fifo_len; s->fifo_len --, sample ++) | |
1257 | + s->codec_out(s->opaque, *sample); | |
1258 | + pxa2xx_i2s_update(s); | |
1259 | +} | |
1260 | + | |
1261 | +static struct pxa2xx_i2s_s *pxa2xx_i2s_init(target_phys_addr_t base, | |
1262 | + qemu_irq irq, struct pxa2xx_dma_state_s *dma) | |
1263 | +{ | |
1264 | + int iomemtype; | |
1265 | + struct pxa2xx_i2s_s *s = (struct pxa2xx_i2s_s *) | |
1266 | + qemu_mallocz(sizeof(struct pxa2xx_i2s_s)); | |
1267 | + | |
1268 | + s->base = base; | |
1269 | + s->irq = irq; | |
1270 | + s->dma = dma; | |
1271 | + s->data_req = pxa2xx_i2s_data_req; | |
1272 | + | |
1273 | + pxa2xx_i2s_reset(s); | |
1274 | + | |
1275 | + iomemtype = cpu_register_io_memory(0, pxa2xx_i2s_readfn, | |
1276 | + pxa2xx_i2s_writefn, s); | |
1277 | + cpu_register_physical_memory(s->base & 0xfff00000, 0xfffff, iomemtype); | |
1278 | + | |
1279 | + return s; | |
1280 | +} | |
1281 | + | |
1282 | +/* PXA Fast Infra-red Communications Port */ | |
1283 | +struct pxa2xx_fir_s { | |
1284 | + target_phys_addr_t base; | |
1285 | + qemu_irq irq; | |
1286 | + struct pxa2xx_dma_state_s *dma; | |
1287 | + int enable; | |
1288 | + CharDriverState *chr; | |
1289 | + | |
1290 | + uint8_t control[3]; | |
1291 | + uint8_t status[2]; | |
1292 | + | |
1293 | + int rx_len; | |
1294 | + int rx_start; | |
1295 | + uint8_t rx_fifo[64]; | |
1296 | +}; | |
1297 | + | |
1298 | +static void pxa2xx_fir_reset(struct pxa2xx_fir_s *s) | |
1299 | +{ | |
1300 | + s->control[0] = 0x00; | |
1301 | + s->control[1] = 0x00; | |
1302 | + s->control[2] = 0x00; | |
1303 | + s->status[0] = 0x00; | |
1304 | + s->status[1] = 0x00; | |
1305 | + s->enable = 0; | |
1306 | +} | |
1307 | + | |
1308 | +static inline void pxa2xx_fir_update(struct pxa2xx_fir_s *s) | |
1309 | +{ | |
1310 | + static const int tresh[4] = { 8, 16, 32, 0 }; | |
1311 | + int intr = 0; | |
1312 | + if ((s->control[0] & (1 << 4)) && /* RXE */ | |
1313 | + s->rx_len >= tresh[s->control[2] & 3]) /* TRIG */ | |
1314 | + s->status[0] |= 1 << 4; /* RFS */ | |
1315 | + else | |
1316 | + s->status[0] &= ~(1 << 4); /* RFS */ | |
1317 | + if (s->control[0] & (1 << 3)) /* TXE */ | |
1318 | + s->status[0] |= 1 << 3; /* TFS */ | |
1319 | + else | |
1320 | + s->status[0] &= ~(1 << 3); /* TFS */ | |
1321 | + if (s->rx_len) | |
1322 | + s->status[1] |= 1 << 2; /* RNE */ | |
1323 | + else | |
1324 | + s->status[1] &= ~(1 << 2); /* RNE */ | |
1325 | + if (s->control[0] & (1 << 4)) /* RXE */ | |
1326 | + s->status[1] |= 1 << 0; /* RSY */ | |
1327 | + else | |
1328 | + s->status[1] &= ~(1 << 0); /* RSY */ | |
1329 | + | |
1330 | + intr |= (s->control[0] & (1 << 5)) && /* RIE */ | |
1331 | + (s->status[0] & (1 << 4)); /* RFS */ | |
1332 | + intr |= (s->control[0] & (1 << 6)) && /* TIE */ | |
1333 | + (s->status[0] & (1 << 3)); /* TFS */ | |
1334 | + intr |= (s->control[2] & (1 << 4)) && /* TRAIL */ | |
1335 | + (s->status[0] & (1 << 6)); /* EOC */ | |
1336 | + intr |= (s->control[0] & (1 << 2)) && /* TUS */ | |
1337 | + (s->status[0] & (1 << 1)); /* TUR */ | |
1338 | + intr |= s->status[0] & 0x25; /* FRE, RAB, EIF */ | |
1339 | + | |
1340 | + pxa2xx_dma_request(s->dma, PXA2XX_RX_RQ_ICP, (s->status[0] >> 4) & 1); | |
1341 | + pxa2xx_dma_request(s->dma, PXA2XX_TX_RQ_ICP, (s->status[0] >> 3) & 1); | |
1342 | + | |
1343 | + qemu_set_irq(s->irq, intr && s->enable); | |
1344 | +} | |
1345 | + | |
1346 | +#define ICCR0 0x00 /* FICP Control register 0 */ | |
1347 | +#define ICCR1 0x04 /* FICP Control register 1 */ | |
1348 | +#define ICCR2 0x08 /* FICP Control register 2 */ | |
1349 | +#define ICDR 0x0c /* FICP Data register */ | |
1350 | +#define ICSR0 0x14 /* FICP Status register 0 */ | |
1351 | +#define ICSR1 0x18 /* FICP Status register 1 */ | |
1352 | +#define ICFOR 0x1c /* FICP FIFO Occupancy Status register */ | |
1353 | + | |
1354 | +static uint32_t pxa2xx_fir_read(void *opaque, target_phys_addr_t addr) | |
1355 | +{ | |
1356 | + struct pxa2xx_fir_s *s = (struct pxa2xx_fir_s *) opaque; | |
1357 | + uint8_t ret; | |
1358 | + addr -= s->base; | |
1359 | + | |
1360 | + switch (addr) { | |
1361 | + case ICCR0: | |
1362 | + return s->control[0]; | |
1363 | + case ICCR1: | |
1364 | + return s->control[1]; | |
1365 | + case ICCR2: | |
1366 | + return s->control[2]; | |
1367 | + case ICDR: | |
1368 | + s->status[0] &= ~0x01; | |
1369 | + s->status[1] &= ~0x72; | |
1370 | + if (s->rx_len) { | |
1371 | + s->rx_len --; | |
1372 | + ret = s->rx_fifo[s->rx_start ++]; | |
1373 | + s->rx_start &= 63; | |
1374 | + pxa2xx_fir_update(s); | |
1375 | + return ret; | |
1376 | + } | |
1377 | + printf("%s: Rx FIFO underrun.\n", __FUNCTION__); | |
1378 | + break; | |
1379 | + case ICSR0: | |
1380 | + return s->status[0]; | |
1381 | + case ICSR1: | |
1382 | + return s->status[1] | (1 << 3); /* TNF */ | |
1383 | + case ICFOR: | |
1384 | + return s->rx_len; | |
1385 | + default: | |
1386 | + printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr); | |
1387 | + break; | |
1388 | + } | |
1389 | + return 0; | |
1390 | +} | |
1391 | + | |
1392 | +static void pxa2xx_fir_write(void *opaque, target_phys_addr_t addr, | |
1393 | + uint32_t value) | |
1394 | +{ | |
1395 | + struct pxa2xx_fir_s *s = (struct pxa2xx_fir_s *) opaque; | |
1396 | + uint8_t ch; | |
1397 | + addr -= s->base; | |
1398 | + | |
1399 | + switch (addr) { | |
1400 | + case ICCR0: | |
1401 | + s->control[0] = value; | |
1402 | + if (!(value & (1 << 4))) /* RXE */ | |
1403 | + s->rx_len = s->rx_start = 0; | |
1404 | + if (!(value & (1 << 3))) /* TXE */ | |
1405 | + /* Nop */; | |
1406 | + s->enable = value & 1; /* ITR */ | |
1407 | + if (!s->enable) | |
1408 | + s->status[0] = 0; | |
1409 | + pxa2xx_fir_update(s); | |
1410 | + break; | |
1411 | + case ICCR1: | |
1412 | + s->control[1] = value; | |
1413 | + break; | |
1414 | + case ICCR2: | |
1415 | + s->control[2] = value & 0x3f; | |
1416 | + pxa2xx_fir_update(s); | |
1417 | + break; | |
1418 | + case ICDR: | |
1419 | + if (s->control[2] & (1 << 2)) /* TXP */ | |
1420 | + ch = value; | |
1421 | + else | |
1422 | + ch = ~value; | |
1423 | + if (s->chr && s->enable && (s->control[0] & (1 << 3))) /* TXE */ | |
1424 | + qemu_chr_write(s->chr, &ch, 1); | |
1425 | + break; | |
1426 | + case ICSR0: | |
1427 | + s->status[0] &= ~(value & 0x66); | |
1428 | + pxa2xx_fir_update(s); | |
1429 | + break; | |
1430 | + case ICFOR: | |
1431 | + break; | |
1432 | + default: | |
1433 | + printf("%s: Bad register " REG_FMT "\n", __FUNCTION__, addr); | |
1434 | + } | |
1435 | +} | |
1436 | + | |
1437 | +static CPUReadMemoryFunc *pxa2xx_fir_readfn[] = { | |
1438 | + pxa2xx_fir_read, | |
1439 | + pxa2xx_fir_read, | |
1440 | + pxa2xx_fir_read, | |
1441 | +}; | |
1442 | + | |
1443 | +static CPUWriteMemoryFunc *pxa2xx_fir_writefn[] = { | |
1444 | + pxa2xx_fir_write, | |
1445 | + pxa2xx_fir_write, | |
1446 | + pxa2xx_fir_write, | |
1447 | +}; | |
1448 | + | |
1449 | +static int pxa2xx_fir_is_empty(void *opaque) | |
1450 | +{ | |
1451 | + struct pxa2xx_fir_s *s = (struct pxa2xx_fir_s *) opaque; | |
1452 | + return (s->rx_len < 64); | |
1453 | +} | |
1454 | + | |
1455 | +static void pxa2xx_fir_rx(void *opaque, const uint8_t *buf, int size) | |
1456 | +{ | |
1457 | + struct pxa2xx_fir_s *s = (struct pxa2xx_fir_s *) opaque; | |
1458 | + if (!(s->control[0] & (1 << 4))) /* RXE */ | |
1459 | + return; | |
1460 | + | |
1461 | + while (size --) { | |
1462 | + s->status[1] |= 1 << 4; /* EOF */ | |
1463 | + if (s->rx_len >= 64) { | |
1464 | + s->status[1] |= 1 << 6; /* ROR */ | |
1465 | + break; | |
1466 | + } | |
1467 | + | |
1468 | + if (s->control[2] & (1 << 3)) /* RXP */ | |
1469 | + s->rx_fifo[(s->rx_start + s->rx_len ++) & 63] = *(buf ++); | |
1470 | + else | |
1471 | + s->rx_fifo[(s->rx_start + s->rx_len ++) & 63] = ~*(buf ++); | |
1472 | + } | |
1473 | + | |
1474 | + pxa2xx_fir_update(s); | |
1475 | +} | |
1476 | + | |
1477 | +static void pxa2xx_fir_event(void *opaque, int event) | |
1478 | +{ | |
1479 | +} | |
1480 | + | |
1481 | +static struct pxa2xx_fir_s *pxa2xx_fir_init(target_phys_addr_t base, | |
1482 | + qemu_irq irq, struct pxa2xx_dma_state_s *dma, | |
1483 | + CharDriverState *chr) | |
1484 | +{ | |
1485 | + int iomemtype; | |
1486 | + struct pxa2xx_fir_s *s = (struct pxa2xx_fir_s *) | |
1487 | + qemu_mallocz(sizeof(struct pxa2xx_fir_s)); | |
1488 | + | |
1489 | + s->base = base; | |
1490 | + s->irq = irq; | |
1491 | + s->dma = dma; | |
1492 | + s->chr = chr; | |
1493 | + | |
1494 | + pxa2xx_fir_reset(s); | |
1495 | + | |
1496 | + iomemtype = cpu_register_io_memory(0, pxa2xx_fir_readfn, | |
1497 | + pxa2xx_fir_writefn, s); | |
1498 | + cpu_register_physical_memory(s->base, 0xfff, iomemtype); | |
1499 | + | |
1500 | + if (chr) | |
1501 | + qemu_chr_add_handlers(chr, pxa2xx_fir_is_empty, | |
1502 | + pxa2xx_fir_rx, pxa2xx_fir_event, s); | |
1503 | + | |
1504 | + return s; | |
1505 | +} | |
1506 | + | |
1507 | +void pxa2xx_reset(int line, int level, void *opaque) | |
1508 | +{ | |
1509 | + struct pxa2xx_state_s *s = (struct pxa2xx_state_s *) opaque; | |
1510 | + if (level && (s->pm_regs[PCFR >> 2] & 0x10)) { /* GPR_EN */ | |
1511 | + cpu_reset(s->env); | |
1512 | + /* TODO: reset peripherals */ | |
1513 | + } | |
1514 | +} | |
1515 | + | |
1516 | +/* Initialise a PXA270 integrated chip (ARM based core). */ | |
1517 | +struct pxa2xx_state_s *pxa270_init(DisplayState *ds, const char *revision) | |
1518 | +{ | |
1519 | + struct pxa2xx_state_s *s; | |
1520 | + struct pxa2xx_ssp_s *ssp; | |
1521 | + char *cpu_model; | |
1522 | + int iomemtype, i; | |
1523 | + s = (struct pxa2xx_state_s *) qemu_mallocz(sizeof(struct pxa2xx_state_s)); | |
1524 | + | |
1525 | + s->env = cpu_init(); | |
1526 | + asprintf(&cpu_model, "pxa270-%s", revision); | |
1527 | + cpu_arm_set_model(s->env, cpu_model); | |
1528 | + free(cpu_model); | |
1529 | + | |
1530 | + s->pic = pxa2xx_pic_init(0x40d00000, s->env); | |
1531 | + | |
1532 | + s->dma = pxa27x_dma_init(0x40000000, s->pic[PXA2XX_PIC_DMA]); | |
1533 | + | |
1534 | + s->gpio = pxa2xx_gpio_init(0x40e00000, s->env, s->pic, 121); | |
1535 | + | |
1536 | + for (i = 0; pxa270_serial[i].io_base; i ++) | |
1537 | + if (serial_hds[i]) | |
1538 | + serial_mm_init(pxa270_serial[i].io_base, 2, | |
1539 | + s->pic[pxa270_serial[i].irqn], serial_hds[i], 1); | |
1540 | + else | |
1541 | + break; | |
1542 | + if (serial_hds[i]) | |
1543 | + s->fir = pxa2xx_fir_init(0x40800000, s->pic[PXA2XX_PIC_ICP], | |
1544 | + s->dma, serial_hds[i]); | |
1545 | + | |
1546 | + s->cm_base = 0x41300000; | |
1547 | + s->cm_regs[CCCR >> 4] = 0x02000210; /* 416.0 MHz */ | |
1548 | + s->clkcfg = 0x00000009; /* Turbo mode active */ | |
1549 | + iomemtype = cpu_register_io_memory(0, pxa2xx_cm_readfn, | |
1550 | + pxa2xx_cm_writefn, s); | |
1551 | + cpu_register_physical_memory(s->cm_base, 0xfff, iomemtype); | |
1552 | + | |
1553 | + cpu_arm_set_cp_io(s->env, 14, pxa2xx_cp14_read, pxa2xx_cp14_write, s); | |
1554 | + | |
1555 | + s->mm_base = 0x48000000; | |
1556 | + s->mm_regs[MDMRS >> 2] = 0x00020002; | |
1557 | + s->mm_regs[MDREFR >> 2] = 0x03ca4000; | |
1558 | + s->mm_regs[MECR >> 2] = 0x00000001; /* Two PC Card sockets */ | |
1559 | + iomemtype = cpu_register_io_memory(0, pxa2xx_mm_readfn, | |
1560 | + pxa2xx_mm_writefn, s); | |
1561 | + cpu_register_physical_memory(s->mm_base, 0xfff, iomemtype); | |
1562 | + | |
1563 | + for (i = 0; pxa27x_ssp[i].io_base; i ++); | |
1564 | + s->ssp = (struct pxa2xx_ssp_s **) | |
1565 | + qemu_mallocz(sizeof(struct pxa2xx_ssp_s *) * i); | |
1566 | + ssp = (struct pxa2xx_ssp_s *) | |
1567 | + qemu_mallocz(sizeof(struct pxa2xx_ssp_s) * i); | |
1568 | + for (i = 0; pxa27x_ssp[i].io_base; i ++) { | |
1569 | + s->ssp[i] = &ssp[i]; | |
1570 | + ssp[i].base = pxa27x_ssp[i].io_base; | |
1571 | + ssp[i].irq = s->pic[pxa27x_ssp[i].irqn]; | |
1572 | + | |
1573 | + iomemtype = cpu_register_io_memory(0, pxa2xx_ssp_readfn, | |
1574 | + pxa2xx_ssp_writefn, &ssp[i]); | |
1575 | + cpu_register_physical_memory(ssp[i].base, 0xfff, iomemtype); | |
1576 | + } | |
1577 | + | |
1578 | + s->rtc_base = 0x40900000; | |
1579 | + iomemtype = cpu_register_io_memory(0, pxa2xx_rtc_readfn, | |
1580 | + pxa2xx_rtc_writefn, s); | |
1581 | + cpu_register_physical_memory(s->rtc_base, 0xfff, iomemtype); | |
1582 | + pxa2xx_rtc_reset(s); | |
1583 | + | |
1584 | + s->pm_base = 0x40f00000; | |
1585 | + iomemtype = cpu_register_io_memory(0, pxa2xx_pm_readfn, | |
1586 | + pxa2xx_pm_writefn, s); | |
1587 | + cpu_register_physical_memory(s->pm_base, 0xfff, iomemtype); | |
1588 | + | |
1589 | + s->i2s = pxa2xx_i2s_init(0x40400000, s->pic[PXA2XX_PIC_I2S], s->dma); | |
1590 | + | |
1591 | + /* GPIO1 resets the processor */ | |
1592 | + /* The handler can be overriden by board-specific code */ | |
1593 | + pxa2xx_gpio_handler_set(s->gpio, 1, pxa2xx_reset, s); | |
1594 | + return s; | |
1595 | +} | |
1596 | + | |
1597 | +/* Initialise a PXA255 integrated chip (ARM based core). */ | |
1598 | +struct pxa2xx_state_s *pxa255_init(DisplayState *ds) | |
1599 | +{ | |
1600 | + struct pxa2xx_state_s *s; | |
1601 | + struct pxa2xx_ssp_s *ssp; | |
1602 | + int iomemtype, i; | |
1603 | + s = (struct pxa2xx_state_s *) qemu_mallocz(sizeof(struct pxa2xx_state_s)); | |
1604 | + | |
1605 | + s->env = cpu_init(); | |
1606 | + cpu_arm_set_model(s->env, "pxa255"); | |
1607 | + | |
1608 | + s->pic = pxa2xx_pic_init(0x40d00000, s->env); | |
1609 | + | |
1610 | + s->dma = pxa255_dma_init(0x40000000, s->pic[PXA2XX_PIC_DMA]); | |
1611 | + | |
1612 | + s->gpio = pxa2xx_gpio_init(0x40e00000, s->env, s->pic, 121); | |
1613 | + | |
1614 | + for (i = 0; pxa255_serial[i].io_base; i ++) | |
1615 | + if (serial_hds[i]) | |
1616 | + serial_mm_init(pxa255_serial[i].io_base, 2, | |
1617 | + s->pic[pxa255_serial[i].irqn], serial_hds[i], 1); | |
1618 | + else | |
1619 | + break; | |
1620 | + if (serial_hds[i]) | |
1621 | + s->fir = pxa2xx_fir_init(0x40800000, s->pic[PXA2XX_PIC_ICP], | |
1622 | + s->dma, serial_hds[i]); | |
1623 | + | |
1624 | + s->cm_base = 0x41300000; | |
1625 | + s->cm_regs[CCCR >> 4] = 0x02000210; /* 416.0 MHz */ | |
1626 | + s->clkcfg = 0x00000009; /* Turbo mode active */ | |
1627 | + iomemtype = cpu_register_io_memory(0, pxa2xx_cm_readfn, | |
1628 | + pxa2xx_cm_writefn, s); | |
1629 | + cpu_register_physical_memory(s->cm_base, 0xfff, iomemtype); | |
1630 | + | |
1631 | + cpu_arm_set_cp_io(s->env, 14, pxa2xx_cp14_read, pxa2xx_cp14_write, s); | |
1632 | + | |
1633 | + s->mm_base = 0x48000000; | |
1634 | + s->mm_regs[MDMRS >> 2] = 0x00020002; | |
1635 | + s->mm_regs[MDREFR >> 2] = 0x03ca4000; | |
1636 | + s->mm_regs[MECR >> 2] = 0x00000001; /* Two PC Card sockets */ | |
1637 | + iomemtype = cpu_register_io_memory(0, pxa2xx_mm_readfn, | |
1638 | + pxa2xx_mm_writefn, s); | |
1639 | + cpu_register_physical_memory(s->mm_base, 0xfff, iomemtype); | |
1640 | + | |
1641 | + for (i = 0; pxa255_ssp[i].io_base; i ++); | |
1642 | + s->ssp = (struct pxa2xx_ssp_s **) | |
1643 | + qemu_mallocz(sizeof(struct pxa2xx_ssp_s *) * i); | |
1644 | + ssp = (struct pxa2xx_ssp_s *) | |
1645 | + qemu_mallocz(sizeof(struct pxa2xx_ssp_s) * i); | |
1646 | + for (i = 0; pxa255_ssp[i].io_base; i ++) { | |
1647 | + s->ssp[i] = &ssp[i]; | |
1648 | + ssp[i].base = pxa255_ssp[i].io_base; | |
1649 | + ssp[i].irq = s->pic[pxa255_ssp[i].irqn]; | |
1650 | + | |
1651 | + iomemtype = cpu_register_io_memory(0, pxa2xx_ssp_readfn, | |
1652 | + pxa2xx_ssp_writefn, &ssp[i]); | |
1653 | + cpu_register_physical_memory(ssp[i].base, 0xfff, iomemtype); | |
1654 | + } | |
1655 | + | |
1656 | + s->rtc_base = 0x40900000; | |
1657 | + iomemtype = cpu_register_io_memory(0, pxa2xx_rtc_readfn, | |
1658 | + pxa2xx_rtc_writefn, s); | |
1659 | + cpu_register_physical_memory(s->rtc_base, 0xfff, iomemtype); | |
1660 | + pxa2xx_rtc_reset(s); | |
1661 | + | |
1662 | + s->pm_base = 0x40f00000; | |
1663 | + iomemtype = cpu_register_io_memory(0, pxa2xx_pm_readfn, | |
1664 | + pxa2xx_pm_writefn, s); | |
1665 | + cpu_register_physical_memory(s->pm_base, 0xfff, iomemtype); | |
1666 | + | |
1667 | + s->i2s = pxa2xx_i2s_init(0x40400000, s->pic[PXA2XX_PIC_I2S], s->dma); | |
1668 | + | |
1669 | + /* GPIO1 resets the processor */ | |
1670 | + /* The handler can be overriden by board-specific code */ | |
1671 | + pxa2xx_gpio_handler_set(s->gpio, 1, pxa2xx_reset, s); | |
1672 | + return s; | |
1673 | +} | ... | ... |
hw/pxa2xx_dma.c
0 → 100644
1 | +/* | |
2 | + * Intel XScale PXA255/270 DMA controller. | |
3 | + * | |
4 | + * Copyright (c) 2006 Openedhand Ltd. | |
5 | + * Copyright (c) 2006 Thorsten Zitterell | |
6 | + * Written by Andrzej Zaborowski <balrog@zabor.org> | |
7 | + * | |
8 | + * This code is licenced under the GPL. | |
9 | + */ | |
10 | + | |
11 | +#include "vl.h" | |
12 | + | |
13 | +struct pxa2xx_dma_channel_s { | |
14 | + target_phys_addr_t descr; | |
15 | + target_phys_addr_t src; | |
16 | + target_phys_addr_t dest; | |
17 | + uint32_t cmd; | |
18 | + uint32_t state; | |
19 | + int request; | |
20 | +}; | |
21 | + | |
22 | +/* Allow the DMA to be used as a PIC. */ | |
23 | +typedef void (*pxa2xx_dma_handler_t)(void *opaque, int irq, int level); | |
24 | + | |
25 | +struct pxa2xx_dma_state_s { | |
26 | + pxa2xx_dma_handler_t handler; | |
27 | + target_phys_addr_t base; | |
28 | + qemu_irq irq; | |
29 | + | |
30 | + uint32_t stopintr; | |
31 | + uint32_t eorintr; | |
32 | + uint32_t rasintr; | |
33 | + uint32_t startintr; | |
34 | + uint32_t endintr; | |
35 | + | |
36 | + uint32_t align; | |
37 | + uint32_t pio; | |
38 | + | |
39 | + int channels; | |
40 | + struct pxa2xx_dma_channel_s *chan; | |
41 | + | |
42 | + uint8_t *req; | |
43 | + | |
44 | + /* Flag to avoid recursive DMA invocations. */ | |
45 | + int running; | |
46 | +}; | |
47 | + | |
48 | +#define PXA255_DMA_NUM_CHANNELS 16 | |
49 | +#define PXA27X_DMA_NUM_CHANNELS 32 | |
50 | + | |
51 | +#define PXA2XX_DMA_NUM_REQUESTS 75 | |
52 | + | |
53 | +#define DCSR0 0x0000 /* DMA Control / Status register for Channel 0 */ | |
54 | +#define DCSR31 0x007c /* DMA Control / Status register for Channel 31 */ | |
55 | +#define DALGN 0x00a0 /* DMA Alignment register */ | |
56 | +#define DPCSR 0x00a4 /* DMA Programmed I/O Control Status register */ | |
57 | +#define DRQSR0 0x00e0 /* DMA DREQ<0> Status register */ | |
58 | +#define DRQSR1 0x00e4 /* DMA DREQ<1> Status register */ | |
59 | +#define DRQSR2 0x00e8 /* DMA DREQ<2> Status register */ | |
60 | +#define DINT 0x00f0 /* DMA Interrupt register */ | |
61 | +#define DRCMR0 0x0100 /* Request to Channel Map register 0 */ | |
62 | +#define DRCMR63 0x01fc /* Request to Channel Map register 63 */ | |
63 | +#define D_CH0 0x0200 /* Channel 0 Descriptor start */ | |
64 | +#define DRCMR64 0x1100 /* Request to Channel Map register 64 */ | |
65 | +#define DRCMR74 0x1128 /* Request to Channel Map register 74 */ | |
66 | + | |
67 | +/* Per-channel register */ | |
68 | +#define DDADR 0x00 | |
69 | +#define DSADR 0x01 | |
70 | +#define DTADR 0x02 | |
71 | +#define DCMD 0x03 | |
72 | + | |
73 | +/* Bit-field masks */ | |
74 | +#define DRCMR_CHLNUM 0x1f | |
75 | +#define DRCMR_MAPVLD (1 << 7) | |
76 | +#define DDADR_STOP (1 << 0) | |
77 | +#define DDADR_BREN (1 << 1) | |
78 | +#define DCMD_LEN 0x1fff | |
79 | +#define DCMD_WIDTH(x) (1 << ((((x) >> 14) & 3) - 1)) | |
80 | +#define DCMD_SIZE(x) (4 << (((x) >> 16) & 3)) | |
81 | +#define DCMD_FLYBYT (1 << 19) | |
82 | +#define DCMD_FLYBYS (1 << 20) | |
83 | +#define DCMD_ENDIRQEN (1 << 21) | |
84 | +#define DCMD_STARTIRQEN (1 << 22) | |
85 | +#define DCMD_CMPEN (1 << 25) | |
86 | +#define DCMD_FLOWTRG (1 << 28) | |
87 | +#define DCMD_FLOWSRC (1 << 29) | |
88 | +#define DCMD_INCTRGADDR (1 << 30) | |
89 | +#define DCMD_INCSRCADDR (1 << 31) | |
90 | +#define DCSR_BUSERRINTR (1 << 0) | |
91 | +#define DCSR_STARTINTR (1 << 1) | |
92 | +#define DCSR_ENDINTR (1 << 2) | |
93 | +#define DCSR_STOPINTR (1 << 3) | |
94 | +#define DCSR_RASINTR (1 << 4) | |
95 | +#define DCSR_REQPEND (1 << 8) | |
96 | +#define DCSR_EORINT (1 << 9) | |
97 | +#define DCSR_CMPST (1 << 10) | |
98 | +#define DCSR_MASKRUN (1 << 22) | |
99 | +#define DCSR_RASIRQEN (1 << 23) | |
100 | +#define DCSR_CLRCMPST (1 << 24) | |
101 | +#define DCSR_SETCMPST (1 << 25) | |
102 | +#define DCSR_EORSTOPEN (1 << 26) | |
103 | +#define DCSR_EORJMPEN (1 << 27) | |
104 | +#define DCSR_EORIRQEN (1 << 28) | |
105 | +#define DCSR_STOPIRQEN (1 << 29) | |
106 | +#define DCSR_NODESCFETCH (1 << 30) | |
107 | +#define DCSR_RUN (1 << 31) | |
108 | + | |
109 | +static inline void pxa2xx_dma_update(struct pxa2xx_dma_state_s *s, int ch) | |
110 | +{ | |
111 | + if (ch >= 0) { | |
112 | + if ((s->chan[ch].state & DCSR_STOPIRQEN) && | |
113 | + (s->chan[ch].state & DCSR_STOPINTR)) | |
114 | + s->stopintr |= 1 << ch; | |
115 | + else | |
116 | + s->stopintr &= ~(1 << ch); | |
117 | + | |
118 | + if ((s->chan[ch].state & DCSR_EORIRQEN) && | |
119 | + (s->chan[ch].state & DCSR_EORINT)) | |
120 | + s->eorintr |= 1 << ch; | |
121 | + else | |
122 | + s->eorintr &= ~(1 << ch); | |
123 | + | |
124 | + if ((s->chan[ch].state & DCSR_RASIRQEN) && | |
125 | + (s->chan[ch].state & DCSR_RASINTR)) | |
126 | + s->rasintr |= 1 << ch; | |
127 | + else | |
128 | + s->rasintr &= ~(1 << ch); | |
129 | + | |
130 | + if (s->chan[ch].state & DCSR_STARTINTR) | |
131 | + s->startintr |= 1 << ch; | |
132 | + else | |
133 | + s->startintr &= ~(1 << ch); | |
134 | + | |
135 | + if (s->chan[ch].state & DCSR_ENDINTR) | |
136 | + s->endintr |= 1 << ch; | |
137 | + else | |
138 | + s->endintr &= ~(1 << ch); | |
139 | + } | |
140 | + | |
141 | + if (s->stopintr | s->eorintr | s->rasintr | s->startintr | s->endintr) | |
142 | + qemu_irq_raise(s->irq); | |
143 | + else | |
144 | + qemu_irq_lower(s->irq); | |
145 | +} | |
146 | + | |
147 | +static inline void pxa2xx_dma_descriptor_fetch( | |
148 | + struct pxa2xx_dma_state_s *s, int ch) | |
149 | +{ | |
150 | + uint32_t desc[4]; | |
151 | + target_phys_addr_t daddr = s->chan[ch].descr & ~0xf; | |
152 | + if ((s->chan[ch].descr & DDADR_BREN) && (s->chan[ch].state & DCSR_CMPST)) | |
153 | + daddr += 32; | |
154 | + | |
155 | + cpu_physical_memory_read(daddr, (uint8_t *) desc, 16); | |
156 | + s->chan[ch].descr = desc[DDADR]; | |
157 | + s->chan[ch].src = desc[DSADR]; | |
158 | + s->chan[ch].dest = desc[DTADR]; | |
159 | + s->chan[ch].cmd = desc[DCMD]; | |
160 | + | |
161 | + if (s->chan[ch].cmd & DCMD_FLOWSRC) | |
162 | + s->chan[ch].src &= ~3; | |
163 | + if (s->chan[ch].cmd & DCMD_FLOWTRG) | |
164 | + s->chan[ch].dest &= ~3; | |
165 | + | |
166 | + if (s->chan[ch].cmd & (DCMD_CMPEN | DCMD_FLYBYS | DCMD_FLYBYT)) | |
167 | + printf("%s: unsupported mode in channel %i\n", __FUNCTION__, ch); | |
168 | + | |
169 | + if (s->chan[ch].cmd & DCMD_STARTIRQEN) | |
170 | + s->chan[ch].state |= DCSR_STARTINTR; | |
171 | +} | |
172 | + | |
173 | +static void pxa2xx_dma_run(struct pxa2xx_dma_state_s *s) | |
174 | +{ | |
175 | + int c, srcinc, destinc; | |
176 | + uint32_t n, size; | |
177 | + uint32_t width; | |
178 | + uint32_t length; | |
179 | + char buffer[32]; | |
180 | + struct pxa2xx_dma_channel_s *ch; | |
181 | + | |
182 | + if (s->running ++) | |
183 | + return; | |
184 | + | |
185 | + while (s->running) { | |
186 | + s->running = 1; | |
187 | + for (c = 0; c < s->channels; c ++) { | |
188 | + ch = &s->chan[c]; | |
189 | + | |
190 | + while ((ch->state & DCSR_RUN) && !(ch->state & DCSR_STOPINTR)) { | |
191 | + /* Test for pending requests */ | |
192 | + if ((ch->cmd & (DCMD_FLOWSRC | DCMD_FLOWTRG)) && !ch->request) | |
193 | + break; | |
194 | + | |
195 | + length = ch->cmd & DCMD_LEN; | |
196 | + size = DCMD_SIZE(ch->cmd); | |
197 | + width = DCMD_WIDTH(ch->cmd); | |
198 | + | |
199 | + srcinc = (ch->cmd & DCMD_INCSRCADDR) ? width : 0; | |
200 | + destinc = (ch->cmd & DCMD_INCTRGADDR) ? width : 0; | |
201 | + | |
202 | + while (length) { | |
203 | + size = MIN(length, size); | |
204 | + | |
205 | + for (n = 0; n < size; n += width) { | |
206 | + cpu_physical_memory_read(ch->src, buffer + n, width); | |
207 | + ch->src += srcinc; | |
208 | + } | |
209 | + | |
210 | + for (n = 0; n < size; n += width) { | |
211 | + cpu_physical_memory_write(ch->dest, buffer + n, width); | |
212 | + ch->dest += destinc; | |
213 | + } | |
214 | + | |
215 | + length -= size; | |
216 | + | |
217 | + if ((ch->cmd & (DCMD_FLOWSRC | DCMD_FLOWTRG)) && | |
218 | + !ch->request) { | |
219 | + ch->state |= DCSR_EORINT; | |
220 | + if (ch->state & DCSR_EORSTOPEN) | |
221 | + ch->state |= DCSR_STOPINTR; | |
222 | + if ((ch->state & DCSR_EORJMPEN) && | |
223 | + !(ch->state & DCSR_NODESCFETCH)) | |
224 | + pxa2xx_dma_descriptor_fetch(s, c); | |
225 | + break; | |
226 | + } | |
227 | + } | |
228 | + | |
229 | + ch->cmd = (ch->cmd & ~DCMD_LEN) | length; | |
230 | + | |
231 | + /* Is the transfer complete now? */ | |
232 | + if (!length) { | |
233 | + if (ch->cmd & DCMD_ENDIRQEN) | |
234 | + ch->state |= DCSR_ENDINTR; | |
235 | + | |
236 | + if ((ch->state & DCSR_NODESCFETCH) || | |
237 | + (ch->descr & DDADR_STOP) || | |
238 | + (ch->state & DCSR_EORSTOPEN)) { | |
239 | + ch->state |= DCSR_STOPINTR; | |
240 | + ch->state &= ~DCSR_RUN; | |
241 | + | |
242 | + break; | |
243 | + } | |
244 | + | |
245 | + ch->state |= DCSR_STOPINTR; | |
246 | + break; | |
247 | + } | |
248 | + } | |
249 | + } | |
250 | + | |
251 | + s->running --; | |
252 | + } | |
253 | +} | |
254 | + | |
255 | +static uint32_t pxa2xx_dma_read(void *opaque, target_phys_addr_t offset) | |
256 | +{ | |
257 | + struct pxa2xx_dma_state_s *s = (struct pxa2xx_dma_state_s *) opaque; | |
258 | + unsigned int channel; | |
259 | + offset -= s->base; | |
260 | + | |
261 | + switch (offset) { | |
262 | + case DRCMR64 ... DRCMR74: | |
263 | + offset -= DRCMR64 - DRCMR0 - (64 << 2); | |
264 | + /* Fall through */ | |
265 | + case DRCMR0 ... DRCMR63: | |
266 | + channel = (offset - DRCMR0) >> 2; | |
267 | + return s->req[channel]; | |
268 | + | |
269 | + case DRQSR0: | |
270 | + case DRQSR1: | |
271 | + case DRQSR2: | |
272 | + return 0; | |
273 | + | |
274 | + case DCSR0 ... DCSR31: | |
275 | + channel = offset >> 2; | |
276 | + if (s->chan[channel].request) | |
277 | + return s->chan[channel].state | DCSR_REQPEND; | |
278 | + return s->chan[channel].state; | |
279 | + | |
280 | + case DINT: | |
281 | + return s->stopintr | s->eorintr | s->rasintr | | |
282 | + s->startintr | s->endintr; | |
283 | + | |
284 | + case DALGN: | |
285 | + return s->align; | |
286 | + | |
287 | + case DPCSR: | |
288 | + return s->pio; | |
289 | + } | |
290 | + | |
291 | + if (offset >= D_CH0 && offset < D_CH0 + (s->channels << 4)) { | |
292 | + channel = (offset - D_CH0) >> 4; | |
293 | + switch ((offset & 0x0f) >> 2) { | |
294 | + case DDADR: | |
295 | + return s->chan[channel].descr; | |
296 | + case DSADR: | |
297 | + return s->chan[channel].src; | |
298 | + case DTADR: | |
299 | + return s->chan[channel].dest; | |
300 | + case DCMD: | |
301 | + return s->chan[channel].cmd; | |
302 | + } | |
303 | + } | |
304 | + | |
305 | + cpu_abort(cpu_single_env, | |
306 | + "%s: Bad offset 0x%04lx\n", __FUNCTION__, offset); | |
307 | + return 7; | |
308 | +} | |
309 | + | |
310 | +static void pxa2xx_dma_write(void *opaque, | |
311 | + target_phys_addr_t offset, uint32_t value) | |
312 | +{ | |
313 | + struct pxa2xx_dma_state_s *s = (struct pxa2xx_dma_state_s *) opaque; | |
314 | + unsigned int channel; | |
315 | + offset -= s->base; | |
316 | + | |
317 | + switch (offset) { | |
318 | + case DRCMR64 ... DRCMR74: | |
319 | + offset -= DRCMR64 - DRCMR0 - (64 << 2); | |
320 | + /* Fall through */ | |
321 | + case DRCMR0 ... DRCMR63: | |
322 | + channel = (offset - DRCMR0) >> 2; | |
323 | + | |
324 | + if (value & DRCMR_MAPVLD) | |
325 | + if ((value & DRCMR_CHLNUM) > s->channels) | |
326 | + cpu_abort(cpu_single_env, "%s: Bad DMA channel %i\n", | |
327 | + __FUNCTION__, value & DRCMR_CHLNUM); | |
328 | + | |
329 | + s->req[channel] = value; | |
330 | + break; | |
331 | + | |
332 | + case DRQSR0: | |
333 | + case DRQSR1: | |
334 | + case DRQSR2: | |
335 | + /* Nothing to do */ | |
336 | + break; | |
337 | + | |
338 | + case DCSR0 ... DCSR31: | |
339 | + channel = offset >> 2; | |
340 | + s->chan[channel].state &= 0x0000071f & ~(value & | |
341 | + (DCSR_EORINT | DCSR_ENDINTR | | |
342 | + DCSR_STARTINTR | DCSR_BUSERRINTR)); | |
343 | + s->chan[channel].state |= value & 0xfc800000; | |
344 | + | |
345 | + if (s->chan[channel].state & DCSR_STOPIRQEN) | |
346 | + s->chan[channel].state &= ~DCSR_STOPINTR; | |
347 | + | |
348 | + if (value & DCSR_NODESCFETCH) { | |
349 | + /* No-descriptor-fetch mode */ | |
350 | + if (value & DCSR_RUN) | |
351 | + pxa2xx_dma_run(s); | |
352 | + } else { | |
353 | + /* Descriptor-fetch mode */ | |
354 | + if (value & DCSR_RUN) { | |
355 | + s->chan[channel].state &= ~DCSR_STOPINTR; | |
356 | + pxa2xx_dma_descriptor_fetch(s, channel); | |
357 | + pxa2xx_dma_run(s); | |
358 | + } | |
359 | + } | |
360 | + | |
361 | + /* Shouldn't matter as our DMA is synchronous. */ | |
362 | + if (!(value & (DCSR_RUN | DCSR_MASKRUN))) | |
363 | + s->chan[channel].state |= DCSR_STOPINTR; | |
364 | + | |
365 | + if (value & DCSR_CLRCMPST) | |
366 | + s->chan[channel].state &= ~DCSR_CMPST; | |
367 | + if (value & DCSR_SETCMPST) | |
368 | + s->chan[channel].state |= DCSR_CMPST; | |
369 | + | |
370 | + pxa2xx_dma_update(s, channel); | |
371 | + break; | |
372 | + | |
373 | + case DALGN: | |
374 | + s->align = value; | |
375 | + break; | |
376 | + | |
377 | + case DPCSR: | |
378 | + s->pio = value & 0x80000001; | |
379 | + break; | |
380 | + | |
381 | + default: | |
382 | + if (offset >= D_CH0 && offset < D_CH0 + (s->channels << 4)) { | |
383 | + channel = (offset - D_CH0) >> 4; | |
384 | + switch ((offset & 0x0f) >> 2) { | |
385 | + case DDADR: | |
386 | + s->chan[channel].descr = value; | |
387 | + break; | |
388 | + case DSADR: | |
389 | + s->chan[channel].src = value; | |
390 | + break; | |
391 | + case DTADR: | |
392 | + s->chan[channel].dest = value; | |
393 | + break; | |
394 | + case DCMD: | |
395 | + s->chan[channel].cmd = value; | |
396 | + break; | |
397 | + default: | |
398 | + goto fail; | |
399 | + } | |
400 | + | |
401 | + break; | |
402 | + } | |
403 | + fail: | |
404 | + cpu_abort(cpu_single_env, "%s: Bad offset 0x%04lx\n", | |
405 | + __FUNCTION__, offset); | |
406 | + } | |
407 | +} | |
408 | + | |
409 | +static uint32_t pxa2xx_dma_readbad(void *opaque, target_phys_addr_t offset) | |
410 | +{ | |
411 | + cpu_abort(cpu_single_env, "%s: Bad access width\n", __FUNCTION__); | |
412 | + return 5; | |
413 | +} | |
414 | + | |
415 | +static void pxa2xx_dma_writebad(void *opaque, | |
416 | + target_phys_addr_t offset, uint32_t value) | |
417 | +{ | |
418 | + cpu_abort(cpu_single_env, "%s: Bad access width\n", __FUNCTION__); | |
419 | +} | |
420 | + | |
421 | +static CPUReadMemoryFunc *pxa2xx_dma_readfn[] = { | |
422 | + pxa2xx_dma_readbad, | |
423 | + pxa2xx_dma_readbad, | |
424 | + pxa2xx_dma_read | |
425 | +}; | |
426 | + | |
427 | +static CPUWriteMemoryFunc *pxa2xx_dma_writefn[] = { | |
428 | + pxa2xx_dma_writebad, | |
429 | + pxa2xx_dma_writebad, | |
430 | + pxa2xx_dma_write | |
431 | +}; | |
432 | + | |
433 | +static struct pxa2xx_dma_state_s *pxa2xx_dma_init(target_phys_addr_t base, | |
434 | + qemu_irq irq, int channels) | |
435 | +{ | |
436 | + int i, iomemtype; | |
437 | + struct pxa2xx_dma_state_s *s; | |
438 | + s = (struct pxa2xx_dma_state_s *) | |
439 | + qemu_mallocz(sizeof(struct pxa2xx_dma_state_s)); | |
440 | + | |
441 | + s->channels = channels; | |
442 | + s->chan = qemu_mallocz(sizeof(struct pxa2xx_dma_channel_s) * s->channels); | |
443 | + s->base = base; | |
444 | + s->irq = irq; | |
445 | + s->handler = (pxa2xx_dma_handler_t) pxa2xx_dma_request; | |
446 | + s->req = qemu_mallocz(sizeof(int) * PXA2XX_DMA_NUM_REQUESTS); | |
447 | + | |
448 | + memset(s->chan, 0, sizeof(struct pxa2xx_dma_channel_s) * s->channels); | |
449 | + for (i = 0; i < s->channels; i ++) | |
450 | + s->chan[i].state = DCSR_STOPINTR; | |
451 | + | |
452 | + memset(s->req, 0, sizeof(int) * PXA2XX_DMA_NUM_REQUESTS); | |
453 | + | |
454 | + iomemtype = cpu_register_io_memory(0, pxa2xx_dma_readfn, | |
455 | + pxa2xx_dma_writefn, s); | |
456 | + cpu_register_physical_memory(base, 0x0000ffff, iomemtype); | |
457 | + | |
458 | + return s; | |
459 | +} | |
460 | + | |
461 | +struct pxa2xx_dma_state_s *pxa27x_dma_init(target_phys_addr_t base, | |
462 | + qemu_irq irq) | |
463 | +{ | |
464 | + return pxa2xx_dma_init(base, irq, PXA27X_DMA_NUM_CHANNELS); | |
465 | +} | |
466 | + | |
467 | +struct pxa2xx_dma_state_s *pxa255_dma_init(target_phys_addr_t base, | |
468 | + qemu_irq irq) | |
469 | +{ | |
470 | + return pxa2xx_dma_init(base, irq, PXA255_DMA_NUM_CHANNELS); | |
471 | +} | |
472 | + | |
473 | +void pxa2xx_dma_request(struct pxa2xx_dma_state_s *s, int req_num, int on) | |
474 | +{ | |
475 | + int ch; | |
476 | + if (req_num < 0 || req_num >= PXA2XX_DMA_NUM_REQUESTS) | |
477 | + cpu_abort(cpu_single_env, | |
478 | + "%s: Bad DMA request %i\n", __FUNCTION__, req_num); | |
479 | + | |
480 | + if (!(s->req[req_num] & DRCMR_MAPVLD)) | |
481 | + return; | |
482 | + ch = s->req[req_num] & DRCMR_CHLNUM; | |
483 | + | |
484 | + if (!s->chan[ch].request && on) | |
485 | + s->chan[ch].state |= DCSR_RASINTR; | |
486 | + else | |
487 | + s->chan[ch].state &= ~DCSR_RASINTR; | |
488 | + if (s->chan[ch].request && !on) | |
489 | + s->chan[ch].state |= DCSR_EORINT; | |
490 | + | |
491 | + s->chan[ch].request = on; | |
492 | + if (on) { | |
493 | + pxa2xx_dma_run(s); | |
494 | + pxa2xx_dma_update(s, ch); | |
495 | + } | |
496 | +} | ... | ... |
hw/pxa2xx_gpio.c
0 → 100644
1 | +/* | |
2 | + * Intel XScale PXA255/270 GPIO controller emulation. | |
3 | + * | |
4 | + * Copyright (c) 2006 Openedhand Ltd. | |
5 | + * Written by Andrzej Zaborowski <balrog@zabor.org> | |
6 | + * | |
7 | + * This code is licensed under the GPL. | |
8 | + */ | |
9 | + | |
10 | +#include "vl.h" | |
11 | + | |
12 | +#define PXA2XX_GPIO_BANKS 4 | |
13 | + | |
14 | +struct pxa2xx_gpio_info_s { | |
15 | + target_phys_addr_t base; | |
16 | + qemu_irq *pic; | |
17 | + int lines; | |
18 | + CPUState *cpu_env; | |
19 | + | |
20 | + /* XXX: GNU C vectors are more suitable */ | |
21 | + uint32_t ilevel[PXA2XX_GPIO_BANKS]; | |
22 | + uint32_t olevel[PXA2XX_GPIO_BANKS]; | |
23 | + uint32_t dir[PXA2XX_GPIO_BANKS]; | |
24 | + uint32_t rising[PXA2XX_GPIO_BANKS]; | |
25 | + uint32_t falling[PXA2XX_GPIO_BANKS]; | |
26 | + uint32_t status[PXA2XX_GPIO_BANKS]; | |
27 | + uint32_t gafr[PXA2XX_GPIO_BANKS * 2]; | |
28 | + | |
29 | + uint32_t prev_level[PXA2XX_GPIO_BANKS]; | |
30 | + struct { | |
31 | + gpio_handler_t fn; | |
32 | + void *opaque; | |
33 | + } handler[PXA2XX_GPIO_BANKS * 32]; | |
34 | + | |
35 | + void (*read_notify)(void *opaque); | |
36 | + void *opaque; | |
37 | +}; | |
38 | + | |
39 | +static struct { | |
40 | + enum { | |
41 | + GPIO_NONE, | |
42 | + GPLR, | |
43 | + GPSR, | |
44 | + GPCR, | |
45 | + GPDR, | |
46 | + GRER, | |
47 | + GFER, | |
48 | + GEDR, | |
49 | + GAFR_L, | |
50 | + GAFR_U, | |
51 | + } reg; | |
52 | + int bank; | |
53 | +} pxa2xx_gpio_regs[0x200] = { | |
54 | + [0 ... 0x1ff] = { GPIO_NONE, 0 }, | |
55 | +#define PXA2XX_REG(reg, a0, a1, a2, a3) \ | |
56 | + [a0] = { reg, 0 }, [a1] = { reg, 1 }, [a2] = { reg, 2 }, [a3] = { reg, 3 }, | |
57 | + | |
58 | + PXA2XX_REG(GPLR, 0x000, 0x004, 0x008, 0x100) | |
59 | + PXA2XX_REG(GPSR, 0x018, 0x01c, 0x020, 0x118) | |
60 | + PXA2XX_REG(GPCR, 0x024, 0x028, 0x02c, 0x124) | |
61 | + PXA2XX_REG(GPDR, 0x00c, 0x010, 0x014, 0x10c) | |
62 | + PXA2XX_REG(GRER, 0x030, 0x034, 0x038, 0x130) | |
63 | + PXA2XX_REG(GFER, 0x03c, 0x040, 0x044, 0x13c) | |
64 | + PXA2XX_REG(GEDR, 0x048, 0x04c, 0x050, 0x148) | |
65 | + PXA2XX_REG(GAFR_L, 0x054, 0x05c, 0x064, 0x06c) | |
66 | + PXA2XX_REG(GAFR_U, 0x058, 0x060, 0x068, 0x070) | |
67 | +}; | |
68 | + | |
69 | +static void pxa2xx_gpio_irq_update(struct pxa2xx_gpio_info_s *s) | |
70 | +{ | |
71 | + if (s->status[0] & (1 << 0)) | |
72 | + qemu_irq_raise(s->pic[PXA2XX_PIC_GPIO_0]); | |
73 | + else | |
74 | + qemu_irq_lower(s->pic[PXA2XX_PIC_GPIO_0]); | |
75 | + | |
76 | + if (s->status[0] & (1 << 1)) | |
77 | + qemu_irq_raise(s->pic[PXA2XX_PIC_GPIO_1]); | |
78 | + else | |
79 | + qemu_irq_lower(s->pic[PXA2XX_PIC_GPIO_1]); | |
80 | + | |
81 | + if ((s->status[0] & ~3) | s->status[1] | s->status[2] | s->status[3]) | |
82 | + qemu_irq_raise(s->pic[PXA2XX_PIC_GPIO_X]); | |
83 | + else | |
84 | + qemu_irq_lower(s->pic[PXA2XX_PIC_GPIO_X]); | |
85 | +} | |
86 | + | |
87 | +/* Bitmap of pins used as standby and sleep wake-up sources. */ | |
88 | +const int pxa2xx_gpio_wake[PXA2XX_GPIO_BANKS] = { | |
89 | + 0x8003fe1b, 0x002001fc, 0xec080000, 0x0012007f, | |
90 | +}; | |
91 | + | |
92 | +void pxa2xx_gpio_set(struct pxa2xx_gpio_info_s *s, int line, int level) | |
93 | +{ | |
94 | + int bank; | |
95 | + uint32_t mask; | |
96 | + | |
97 | + if (line >= s->lines) { | |
98 | + printf("%s: No GPIO pin %i\n", __FUNCTION__, line); | |
99 | + return; | |
100 | + } | |
101 | + | |
102 | + bank = line >> 5; | |
103 | + mask = 1 << (line & 31); | |
104 | + | |
105 | + if (level) { | |
106 | + s->status[bank] |= s->rising[bank] & mask & | |
107 | + ~s->ilevel[bank] & ~s->dir[bank]; | |
108 | + s->ilevel[bank] |= mask; | |
109 | + } else { | |
110 | + s->status[bank] |= s->falling[bank] & mask & | |
111 | + s->ilevel[bank] & ~s->dir[bank]; | |
112 | + s->ilevel[bank] &= ~mask; | |
113 | + } | |
114 | + | |
115 | + if (s->status[bank] & mask) | |
116 | + pxa2xx_gpio_irq_update(s); | |
117 | + | |
118 | + /* Wake-up GPIOs */ | |
119 | + if (s->cpu_env->halted && (mask & ~s->dir[bank] & pxa2xx_gpio_wake[bank])) | |
120 | + cpu_interrupt(s->cpu_env, CPU_INTERRUPT_EXITTB); | |
121 | +} | |
122 | + | |
123 | +static void pxa2xx_gpio_handler_update(struct pxa2xx_gpio_info_s *s) { | |
124 | + uint32_t level, diff; | |
125 | + int i, bit, line; | |
126 | + for (i = 0; i < PXA2XX_GPIO_BANKS; i ++) { | |
127 | + level = s->olevel[i] & s->dir[i]; | |
128 | + | |
129 | + for (diff = s->prev_level[i] ^ level; diff; diff ^= 1 << bit) { | |
130 | + bit = ffs(diff) - 1; | |
131 | + line = bit + 32 * i; | |
132 | + if (s->handler[line].fn) | |
133 | + s->handler[line].fn(line, (level >> bit) & 1, | |
134 | + s->handler[line].opaque); | |
135 | + } | |
136 | + | |
137 | + s->prev_level[i] = level; | |
138 | + } | |
139 | +} | |
140 | + | |
141 | +static uint32_t pxa2xx_gpio_read(void *opaque, target_phys_addr_t offset) | |
142 | +{ | |
143 | + struct pxa2xx_gpio_info_s *s = (struct pxa2xx_gpio_info_s *) opaque; | |
144 | + uint32_t ret; | |
145 | + int bank; | |
146 | + offset -= s->base; | |
147 | + if (offset >= 0x200) | |
148 | + return 0; | |
149 | + | |
150 | + bank = pxa2xx_gpio_regs[offset].bank; | |
151 | + switch (pxa2xx_gpio_regs[offset].reg) { | |
152 | + case GPDR: /* GPIO Pin-Direction registers */ | |
153 | + return s->dir[bank]; | |
154 | + | |
155 | + case GRER: /* GPIO Rising-Edge Detect Enable registers */ | |
156 | + return s->rising[bank]; | |
157 | + | |
158 | + case GFER: /* GPIO Falling-Edge Detect Enable registers */ | |
159 | + return s->falling[bank]; | |
160 | + | |
161 | + case GAFR_L: /* GPIO Alternate Function registers */ | |
162 | + return s->gafr[bank * 2]; | |
163 | + | |
164 | + case GAFR_U: /* GPIO Alternate Function registers */ | |
165 | + return s->gafr[bank * 2 + 1]; | |
166 | + | |
167 | + case GPLR: /* GPIO Pin-Level registers */ | |
168 | + ret = (s->olevel[bank] & s->dir[bank]) | | |
169 | + (s->ilevel[bank] & ~s->dir[bank]); | |
170 | + if (s->read_notify) | |
171 | + s->read_notify(s->opaque); | |
172 | + return ret; | |
173 | + | |
174 | + case GEDR: /* GPIO Edge Detect Status registers */ | |
175 | + return s->status[bank]; | |
176 | + | |
177 | + default: | |
178 | + cpu_abort(cpu_single_env, | |
179 | + "%s: Bad offset " REG_FMT "\n", __FUNCTION__, offset); | |
180 | + } | |
181 | + | |
182 | + return 0; | |
183 | +} | |
184 | + | |
185 | +static void pxa2xx_gpio_write(void *opaque, | |
186 | + target_phys_addr_t offset, uint32_t value) | |
187 | +{ | |
188 | + struct pxa2xx_gpio_info_s *s = (struct pxa2xx_gpio_info_s *) opaque; | |
189 | + int bank; | |
190 | + offset -= s->base; | |
191 | + if (offset >= 0x200) | |
192 | + return; | |
193 | + | |
194 | + bank = pxa2xx_gpio_regs[offset].bank; | |
195 | + switch (pxa2xx_gpio_regs[offset].reg) { | |
196 | + case GPDR: /* GPIO Pin-Direction registers */ | |
197 | + s->dir[bank] = value; | |
198 | + pxa2xx_gpio_handler_update(s); | |
199 | + break; | |
200 | + | |
201 | + case GPSR: /* GPIO Pin-Output Set registers */ | |
202 | + s->olevel[bank] |= value; | |
203 | + pxa2xx_gpio_handler_update(s); | |
204 | + break; | |
205 | + | |
206 | + case GPCR: /* GPIO Pin-Output Clear registers */ | |
207 | + s->olevel[bank] &= ~value; | |
208 | + pxa2xx_gpio_handler_update(s); | |
209 | + break; | |
210 | + | |
211 | + case GRER: /* GPIO Rising-Edge Detect Enable registers */ | |
212 | + s->rising[bank] = value; | |
213 | + break; | |
214 | + | |
215 | + case GFER: /* GPIO Falling-Edge Detect Enable registers */ | |
216 | + s->falling[bank] = value; | |
217 | + break; | |
218 | + | |
219 | + case GAFR_L: /* GPIO Alternate Function registers */ | |
220 | + s->gafr[bank * 2] = value; | |
221 | + break; | |
222 | + | |
223 | + case GAFR_U: /* GPIO Alternate Function registers */ | |
224 | + s->gafr[bank * 2 + 1] = value; | |
225 | + break; | |
226 | + | |
227 | + case GEDR: /* GPIO Edge Detect Status registers */ | |
228 | + s->status[bank] &= ~value; | |
229 | + pxa2xx_gpio_irq_update(s); | |
230 | + break; | |
231 | + | |
232 | + default: | |
233 | + cpu_abort(cpu_single_env, | |
234 | + "%s: Bad offset " REG_FMT "\n", __FUNCTION__, offset); | |
235 | + } | |
236 | +} | |
237 | + | |
238 | +static CPUReadMemoryFunc *pxa2xx_gpio_readfn[] = { | |
239 | + pxa2xx_gpio_read, | |
240 | + pxa2xx_gpio_read, | |
241 | + pxa2xx_gpio_read | |
242 | +}; | |
243 | + | |
244 | +static CPUWriteMemoryFunc *pxa2xx_gpio_writefn[] = { | |
245 | + pxa2xx_gpio_write, | |
246 | + pxa2xx_gpio_write, | |
247 | + pxa2xx_gpio_write | |
248 | +}; | |
249 | + | |
250 | +struct pxa2xx_gpio_info_s *pxa2xx_gpio_init(target_phys_addr_t base, | |
251 | + CPUState *env, qemu_irq *pic, int lines) | |
252 | +{ | |
253 | + int iomemtype; | |
254 | + struct pxa2xx_gpio_info_s *s; | |
255 | + | |
256 | + s = (struct pxa2xx_gpio_info_s *) | |
257 | + qemu_mallocz(sizeof(struct pxa2xx_gpio_info_s)); | |
258 | + memset(s, 0, sizeof(struct pxa2xx_gpio_info_s)); | |
259 | + s->base = base; | |
260 | + s->pic = pic; | |
261 | + s->lines = lines; | |
262 | + s->cpu_env = env; | |
263 | + | |
264 | + iomemtype = cpu_register_io_memory(0, pxa2xx_gpio_readfn, | |
265 | + pxa2xx_gpio_writefn, s); | |
266 | + cpu_register_physical_memory(base, 0x00000fff, iomemtype); | |
267 | + | |
268 | + return s; | |
269 | +} | |
270 | + | |
271 | +void pxa2xx_gpio_handler_set(struct pxa2xx_gpio_info_s *s, int line, | |
272 | + gpio_handler_t handler, void *opaque) { | |
273 | + if (line >= s->lines) { | |
274 | + printf("%s: No GPIO pin %i\n", __FUNCTION__, line); | |
275 | + return; | |
276 | + } | |
277 | + | |
278 | + s->handler[line].fn = handler; | |
279 | + s->handler[line].opaque = opaque; | |
280 | +} | |
281 | + | |
282 | +/* | |
283 | + * Registers a callback to notify on GPLR reads. This normally | |
284 | + * shouldn't be needed but it is used for the hack on Spitz machines. | |
285 | + */ | |
286 | +void pxa2xx_gpio_read_notifier(struct pxa2xx_gpio_info_s *s, | |
287 | + void (*handler)(void *opaque), void *opaque) { | |
288 | + s->read_notify = handler; | |
289 | + s->opaque = opaque; | |
290 | +} | ... | ... |
hw/pxa2xx_pic.c
0 → 100644
1 | +/* | |
2 | + * Intel XScale PXA Programmable Interrupt Controller. | |
3 | + * | |
4 | + * Copyright (c) 2006 Openedhand Ltd. | |
5 | + * Copyright (c) 2006 Thorsten Zitterell | |
6 | + * Written by Andrzej Zaborowski <balrog@zabor.org> | |
7 | + * | |
8 | + * This code is licenced under the GPL. | |
9 | + */ | |
10 | + | |
11 | +#include "vl.h" | |
12 | + | |
13 | +#define ICIP 0x00 /* Interrupt Controller IRQ Pending register */ | |
14 | +#define ICMR 0x04 /* Interrupt Controller Mask register */ | |
15 | +#define ICLR 0x08 /* Interrupt Controller Level register */ | |
16 | +#define ICFP 0x0c /* Interrupt Controller FIQ Pending register */ | |
17 | +#define ICPR 0x10 /* Interrupt Controller Pending register */ | |
18 | +#define ICCR 0x14 /* Interrupt Controller Control register */ | |
19 | +#define ICHP 0x18 /* Interrupt Controller Highest Priority register */ | |
20 | +#define IPR0 0x1c /* Interrupt Controller Priority register 0 */ | |
21 | +#define IPR31 0x98 /* Interrupt Controller Priority register 31 */ | |
22 | +#define ICIP2 0x9c /* Interrupt Controller IRQ Pending register 2 */ | |
23 | +#define ICMR2 0xa0 /* Interrupt Controller Mask register 2 */ | |
24 | +#define ICLR2 0xa4 /* Interrupt Controller Level register 2 */ | |
25 | +#define ICFP2 0xa8 /* Interrupt Controller FIQ Pending register 2 */ | |
26 | +#define ICPR2 0xac /* Interrupt Controller Pending register 2 */ | |
27 | +#define IPR32 0xb0 /* Interrupt Controller Priority register 32 */ | |
28 | +#define IPR39 0xcc /* Interrupt Controller Priority register 39 */ | |
29 | + | |
30 | +#define PXA2XX_PIC_SRCS 40 | |
31 | + | |
32 | +struct pxa2xx_pic_state_s { | |
33 | + target_phys_addr_t base; | |
34 | + CPUState *cpu_env; | |
35 | + uint32_t int_enabled[2]; | |
36 | + uint32_t int_pending[2]; | |
37 | + uint32_t is_fiq[2]; | |
38 | + uint32_t int_idle; | |
39 | + uint32_t priority[PXA2XX_PIC_SRCS]; | |
40 | +}; | |
41 | + | |
42 | +static void pxa2xx_pic_update(void *opaque) | |
43 | +{ | |
44 | + uint32_t mask[2]; | |
45 | + struct pxa2xx_pic_state_s *s = (struct pxa2xx_pic_state_s *) opaque; | |
46 | + | |
47 | + if (s->cpu_env->halted) { | |
48 | + mask[0] = s->int_pending[0] & (s->int_enabled[0] | s->int_idle); | |
49 | + mask[1] = s->int_pending[1] & (s->int_enabled[1] | s->int_idle); | |
50 | + if (mask[0] || mask[1]) | |
51 | + cpu_interrupt(s->cpu_env, CPU_INTERRUPT_EXITTB); | |
52 | + } | |
53 | + | |
54 | + mask[0] = s->int_pending[0] & s->int_enabled[0]; | |
55 | + mask[1] = s->int_pending[1] & s->int_enabled[1]; | |
56 | + | |
57 | + if ((mask[0] & s->is_fiq[0]) || (mask[1] & s->is_fiq[1])) | |
58 | + cpu_interrupt(s->cpu_env, CPU_INTERRUPT_FIQ); | |
59 | + else | |
60 | + cpu_reset_interrupt(s->cpu_env, CPU_INTERRUPT_FIQ); | |
61 | + | |
62 | + if ((mask[0] & ~s->is_fiq[0]) || (mask[1] & ~s->is_fiq[1])) | |
63 | + cpu_interrupt(s->cpu_env, CPU_INTERRUPT_HARD); | |
64 | + else | |
65 | + cpu_reset_interrupt(s->cpu_env, CPU_INTERRUPT_HARD); | |
66 | +} | |
67 | + | |
68 | +/* Note: Here level means state of the signal on a pin, not | |
69 | + * IRQ/FIQ distinction as in PXA Developer Manual. */ | |
70 | +static void pxa2xx_pic_set_irq(void *opaque, int irq, int level) | |
71 | +{ | |
72 | + struct pxa2xx_pic_state_s *s = (struct pxa2xx_pic_state_s *) opaque; | |
73 | + int int_set = (irq >= 32); | |
74 | + irq &= 31; | |
75 | + | |
76 | + if (level) | |
77 | + s->int_pending[int_set] |= 1 << irq; | |
78 | + else | |
79 | + s->int_pending[int_set] &= ~(1 << irq); | |
80 | + | |
81 | + pxa2xx_pic_update(opaque); | |
82 | +} | |
83 | + | |
84 | +static inline uint32_t pxa2xx_pic_highest(struct pxa2xx_pic_state_s *s) { | |
85 | + int i, int_set, irq; | |
86 | + uint32_t bit, mask[2]; | |
87 | + uint32_t ichp = 0x003f003f; /* Both IDs invalid */ | |
88 | + | |
89 | + mask[0] = s->int_pending[0] & s->int_enabled[0]; | |
90 | + mask[1] = s->int_pending[1] & s->int_enabled[1]; | |
91 | + | |
92 | + for (i = PXA2XX_PIC_SRCS - 1; i >= 0; i --) { | |
93 | + irq = s->priority[i] & 0x3f; | |
94 | + if ((s->priority[i] & (1 << 31)) && irq < PXA2XX_PIC_SRCS) { | |
95 | + /* Source peripheral ID is valid. */ | |
96 | + bit = 1 << (irq & 31); | |
97 | + int_set = (irq >= 32); | |
98 | + | |
99 | + if (mask[int_set] & bit & s->is_fiq[int_set]) { | |
100 | + /* FIQ asserted */ | |
101 | + ichp &= 0xffff0000; | |
102 | + ichp |= (1 << 15) | irq; | |
103 | + } | |
104 | + | |
105 | + if (mask[int_set] & bit & ~s->is_fiq[int_set]) { | |
106 | + /* IRQ asserted */ | |
107 | + ichp &= 0x0000ffff; | |
108 | + ichp |= (1 << 31) | (irq << 16); | |
109 | + } | |
110 | + } | |
111 | + } | |
112 | + | |
113 | + return ichp; | |
114 | +} | |
115 | + | |
116 | +static uint32_t pxa2xx_pic_mem_read(void *opaque, target_phys_addr_t offset) | |
117 | +{ | |
118 | + struct pxa2xx_pic_state_s *s = (struct pxa2xx_pic_state_s *) opaque; | |
119 | + offset -= s->base; | |
120 | + | |
121 | + switch (offset) { | |
122 | + case ICIP: /* IRQ Pending register */ | |
123 | + return s->int_pending[0] & ~s->is_fiq[0] & s->int_enabled[0]; | |
124 | + case ICIP2: /* IRQ Pending register 2 */ | |
125 | + return s->int_pending[1] & ~s->is_fiq[1] & s->int_enabled[1]; | |
126 | + case ICMR: /* Mask register */ | |
127 | + return s->int_enabled[0]; | |
128 | + case ICMR2: /* Mask register 2 */ | |
129 | + return s->int_enabled[1]; | |
130 | + case ICLR: /* Level register */ | |
131 | + return s->is_fiq[0]; | |
132 | + case ICLR2: /* Level register 2 */ | |
133 | + return s->is_fiq[1]; | |
134 | + case ICCR: /* Idle mask */ | |
135 | + return (s->int_idle == 0); | |
136 | + case ICFP: /* FIQ Pending register */ | |
137 | + return s->int_pending[0] & s->is_fiq[0] & s->int_enabled[0]; | |
138 | + case ICFP2: /* FIQ Pending register 2 */ | |
139 | + return s->int_pending[1] & s->is_fiq[1] & s->int_enabled[1]; | |
140 | + case ICPR: /* Pending register */ | |
141 | + return s->int_pending[0]; | |
142 | + case ICPR2: /* Pending register 2 */ | |
143 | + return s->int_pending[1]; | |
144 | + case IPR0 ... IPR31: | |
145 | + return s->priority[0 + ((offset - IPR0 ) >> 2)]; | |
146 | + case IPR32 ... IPR39: | |
147 | + return s->priority[32 + ((offset - IPR32) >> 2)]; | |
148 | + case ICHP: /* Highest Priority register */ | |
149 | + return pxa2xx_pic_highest(s); | |
150 | + default: | |
151 | + printf("%s: Bad register offset " REG_FMT "\n", __FUNCTION__, offset); | |
152 | + return 0; | |
153 | + } | |
154 | +} | |
155 | + | |
156 | +static void pxa2xx_pic_mem_write(void *opaque, target_phys_addr_t offset, | |
157 | + uint32_t value) | |
158 | +{ | |
159 | + struct pxa2xx_pic_state_s *s = (struct pxa2xx_pic_state_s *) opaque; | |
160 | + offset -= s->base; | |
161 | + | |
162 | + switch (offset) { | |
163 | + case ICMR: /* Mask register */ | |
164 | + s->int_enabled[0] = value; | |
165 | + break; | |
166 | + case ICMR2: /* Mask register 2 */ | |
167 | + s->int_enabled[1] = value; | |
168 | + break; | |
169 | + case ICLR: /* Level register */ | |
170 | + s->is_fiq[0] = value; | |
171 | + break; | |
172 | + case ICLR2: /* Level register 2 */ | |
173 | + s->is_fiq[1] = value; | |
174 | + break; | |
175 | + case ICCR: /* Idle mask */ | |
176 | + s->int_idle = (value & 1) ? 0 : ~0; | |
177 | + break; | |
178 | + case IPR0 ... IPR31: | |
179 | + s->priority[0 + ((offset - IPR0 ) >> 2)] = value & 0x8000003f; | |
180 | + break; | |
181 | + case IPR32 ... IPR39: | |
182 | + s->priority[32 + ((offset - IPR32) >> 2)] = value & 0x8000003f; | |
183 | + break; | |
184 | + default: | |
185 | + printf("%s: Bad register offset " REG_FMT "\n", __FUNCTION__, offset); | |
186 | + return; | |
187 | + } | |
188 | + pxa2xx_pic_update(opaque); | |
189 | +} | |
190 | + | |
191 | +/* Interrupt Controller Coprocessor Space Register Mapping */ | |
192 | +static const int pxa2xx_cp_reg_map[0x10] = { | |
193 | + [0x0 ... 0xf] = -1, | |
194 | + [0x0] = ICIP, | |
195 | + [0x1] = ICMR, | |
196 | + [0x2] = ICLR, | |
197 | + [0x3] = ICFP, | |
198 | + [0x4] = ICPR, | |
199 | + [0x5] = ICHP, | |
200 | + [0x6] = ICIP2, | |
201 | + [0x7] = ICMR2, | |
202 | + [0x8] = ICLR2, | |
203 | + [0x9] = ICFP2, | |
204 | + [0xa] = ICPR2, | |
205 | +}; | |
206 | + | |
207 | +static uint32_t pxa2xx_pic_cp_read(void *opaque, int op2, int reg, int crm) | |
208 | +{ | |
209 | + struct pxa2xx_pic_state_s *s = (struct pxa2xx_pic_state_s *) opaque; | |
210 | + target_phys_addr_t offset; | |
211 | + | |
212 | + if (pxa2xx_cp_reg_map[reg] == -1) { | |
213 | + printf("%s: Bad register 0x%x\n", __FUNCTION__, reg); | |
214 | + return 0; | |
215 | + } | |
216 | + | |
217 | + offset = s->base + pxa2xx_cp_reg_map[reg]; | |
218 | + return pxa2xx_pic_mem_read(opaque, offset); | |
219 | +} | |
220 | + | |
221 | +static void pxa2xx_pic_cp_write(void *opaque, int op2, int reg, int crm, | |
222 | + uint32_t value) | |
223 | +{ | |
224 | + struct pxa2xx_pic_state_s *s = (struct pxa2xx_pic_state_s *) opaque; | |
225 | + target_phys_addr_t offset; | |
226 | + | |
227 | + if (pxa2xx_cp_reg_map[reg] == -1) { | |
228 | + printf("%s: Bad register 0x%x\n", __FUNCTION__, reg); | |
229 | + return; | |
230 | + } | |
231 | + | |
232 | + offset = s->base + pxa2xx_cp_reg_map[reg]; | |
233 | + pxa2xx_pic_mem_write(opaque, offset, value); | |
234 | +} | |
235 | + | |
236 | +static CPUReadMemoryFunc *pxa2xx_pic_readfn[] = { | |
237 | + pxa2xx_pic_mem_read, | |
238 | + pxa2xx_pic_mem_read, | |
239 | + pxa2xx_pic_mem_read, | |
240 | +}; | |
241 | + | |
242 | +static CPUWriteMemoryFunc *pxa2xx_pic_writefn[] = { | |
243 | + pxa2xx_pic_mem_write, | |
244 | + pxa2xx_pic_mem_write, | |
245 | + pxa2xx_pic_mem_write, | |
246 | +}; | |
247 | + | |
248 | +qemu_irq *pxa2xx_pic_init(target_phys_addr_t base, CPUState *env) | |
249 | +{ | |
250 | + struct pxa2xx_pic_state_s *s; | |
251 | + int iomemtype; | |
252 | + qemu_irq *qi; | |
253 | + | |
254 | + s = (struct pxa2xx_pic_state_s *) | |
255 | + qemu_mallocz(sizeof(struct pxa2xx_pic_state_s)); | |
256 | + if (!s) | |
257 | + return NULL; | |
258 | + | |
259 | + s->cpu_env = env; | |
260 | + s->base = base; | |
261 | + | |
262 | + s->int_pending[0] = 0; | |
263 | + s->int_pending[1] = 0; | |
264 | + s->int_enabled[0] = 0; | |
265 | + s->int_enabled[1] = 0; | |
266 | + s->is_fiq[0] = 0; | |
267 | + s->is_fiq[1] = 0; | |
268 | + | |
269 | + qi = qemu_allocate_irqs(pxa2xx_pic_set_irq, s, PXA2XX_PIC_SRCS); | |
270 | + | |
271 | + /* Enable IC memory-mapped registers access. */ | |
272 | + iomemtype = cpu_register_io_memory(0, pxa2xx_pic_readfn, | |
273 | + pxa2xx_pic_writefn, s); | |
274 | + cpu_register_physical_memory(base, 0x000fffff, iomemtype); | |
275 | + | |
276 | + /* Enable IC coprocessor access. */ | |
277 | + cpu_arm_set_cp_io(env, 6, pxa2xx_pic_cp_read, pxa2xx_pic_cp_write, s); | |
278 | + | |
279 | + return qi; | |
280 | +} | ... | ... |
target-arm/cpu.h
... | ... | @@ -38,6 +38,11 @@ |
38 | 38 | #define EXCP_FIQ 6 |
39 | 39 | #define EXCP_BKPT 7 |
40 | 40 | |
41 | +typedef void ARMWriteCPFunc(void *opaque, int cp_info, | |
42 | + int srcreg, int operand, uint32_t value); | |
43 | +typedef uint32_t ARMReadCPFunc(void *opaque, int cp_info, | |
44 | + int dstreg, int operand); | |
45 | + | |
41 | 46 | /* We currently assume float and double are IEEE single and double |
42 | 47 | precision respectively. |
43 | 48 | Doing runtime conversions is tricky because VFP registers may contain |
... | ... | @@ -75,6 +80,7 @@ typedef struct CPUARMState { |
75 | 80 | /* System control coprocessor (cp15) */ |
76 | 81 | struct { |
77 | 82 | uint32_t c0_cpuid; |
83 | + uint32_t c0_cachetype; | |
78 | 84 | uint32_t c1_sys; /* System control register. */ |
79 | 85 | uint32_t c1_coproc; /* Coprocessor access register. */ |
80 | 86 | uint32_t c2; /* MMU translation table base. */ |
... | ... | @@ -87,8 +93,16 @@ typedef struct CPUARMState { |
87 | 93 | uint32_t c9_data; |
88 | 94 | uint32_t c13_fcse; /* FCSE PID. */ |
89 | 95 | uint32_t c13_context; /* Context ID. */ |
96 | + uint32_t c15_cpar; /* XScale Coprocessor Access Register */ | |
90 | 97 | } cp15; |
91 | 98 | |
99 | + /* Coprocessor IO used by peripherals */ | |
100 | + struct { | |
101 | + ARMReadCPFunc *cp_read; | |
102 | + ARMWriteCPFunc *cp_write; | |
103 | + void *opaque; | |
104 | + } cp[15]; | |
105 | + | |
92 | 106 | /* Internal CPU feature flags. */ |
93 | 107 | uint32_t features; |
94 | 108 | |
... | ... | @@ -204,10 +218,10 @@ enum arm_cpu_mode { |
204 | 218 | #define ARM_VFP_FPINST 9 |
205 | 219 | #define ARM_VFP_FPINST2 10 |
206 | 220 | |
207 | - | |
208 | 221 | enum arm_features { |
209 | 222 | ARM_FEATURE_VFP, |
210 | - ARM_FEATURE_AUXCR /* ARM1026 Auxiliary control register. */ | |
223 | + ARM_FEATURE_AUXCR, /* ARM1026 Auxiliary control register. */ | |
224 | + ARM_FEATURE_XSCALE, /* Intel XScale extensions. */ | |
211 | 225 | }; |
212 | 226 | |
213 | 227 | static inline int arm_feature(CPUARMState *env, int feature) |
... | ... | @@ -218,8 +232,24 @@ static inline int arm_feature(CPUARMState *env, int feature) |
218 | 232 | void arm_cpu_list(void); |
219 | 233 | void cpu_arm_set_model(CPUARMState *env, const char *name); |
220 | 234 | |
221 | -#define ARM_CPUID_ARM1026 0x4106a262 | |
222 | -#define ARM_CPUID_ARM926 0x41069265 | |
235 | +void cpu_arm_set_cp_io(CPUARMState *env, int cpnum, | |
236 | + ARMReadCPFunc *cp_read, ARMWriteCPFunc *cp_write, | |
237 | + void *opaque); | |
238 | + | |
239 | +#define ARM_CPUID_ARM1026 0x4106a262 | |
240 | +#define ARM_CPUID_ARM926 0x41069265 | |
241 | +#define ARM_CPUID_PXA250 0x69052100 | |
242 | +#define ARM_CPUID_PXA255 0x69052d00 | |
243 | +#define ARM_CPUID_PXA260 0x69052903 | |
244 | +#define ARM_CPUID_PXA261 0x69052d05 | |
245 | +#define ARM_CPUID_PXA262 0x69052d06 | |
246 | +#define ARM_CPUID_PXA270 0x69054110 | |
247 | +#define ARM_CPUID_PXA270_A0 0x69054110 | |
248 | +#define ARM_CPUID_PXA270_A1 0x69054111 | |
249 | +#define ARM_CPUID_PXA270_B0 0x69054112 | |
250 | +#define ARM_CPUID_PXA270_B1 0x69054113 | |
251 | +#define ARM_CPUID_PXA270_C0 0x69054114 | |
252 | +#define ARM_CPUID_PXA270_C5 0x69054117 | |
223 | 253 | |
224 | 254 | #if defined(CONFIG_USER_ONLY) |
225 | 255 | #define TARGET_PAGE_BITS 12 | ... | ... |
target-arm/exec.h
... | ... | @@ -54,6 +54,8 @@ int cpu_arm_handle_mmu_fault (CPUState *env, target_ulong address, int rw, |
54 | 54 | |
55 | 55 | void cpu_lock(void); |
56 | 56 | void cpu_unlock(void); |
57 | +void helper_set_cp(CPUState *, uint32_t, uint32_t); | |
58 | +uint32_t helper_get_cp(CPUState *, uint32_t); | |
57 | 59 | void helper_set_cp15(CPUState *, uint32_t, uint32_t); |
58 | 60 | uint32_t helper_get_cp15(CPUState *, uint32_t); |
59 | 61 | ... | ... |
target-arm/helper.c
... | ... | @@ -17,11 +17,32 @@ static void cpu_reset_model_id(CPUARMState *env, uint32_t id) |
17 | 17 | case ARM_CPUID_ARM926: |
18 | 18 | set_feature(env, ARM_FEATURE_VFP); |
19 | 19 | env->vfp.xregs[ARM_VFP_FPSID] = 0x41011090; |
20 | + env->cp15.c0_cachetype = 0x1dd20d2; | |
20 | 21 | break; |
21 | 22 | case ARM_CPUID_ARM1026: |
22 | 23 | set_feature(env, ARM_FEATURE_VFP); |
23 | 24 | set_feature(env, ARM_FEATURE_AUXCR); |
24 | 25 | env->vfp.xregs[ARM_VFP_FPSID] = 0x410110a0; |
26 | + env->cp15.c0_cachetype = 0x1dd20d2; | |
27 | + break; | |
28 | + case ARM_CPUID_PXA250: | |
29 | + case ARM_CPUID_PXA255: | |
30 | + case ARM_CPUID_PXA260: | |
31 | + case ARM_CPUID_PXA261: | |
32 | + case ARM_CPUID_PXA262: | |
33 | + set_feature(env, ARM_FEATURE_XSCALE); | |
34 | + /* JTAG_ID is ((id << 28) | 0x09265013) */ | |
35 | + env->cp15.c0_cachetype = 0xd172172; | |
36 | + break; | |
37 | + case ARM_CPUID_PXA270_A0: | |
38 | + case ARM_CPUID_PXA270_A1: | |
39 | + case ARM_CPUID_PXA270_B0: | |
40 | + case ARM_CPUID_PXA270_B1: | |
41 | + case ARM_CPUID_PXA270_C0: | |
42 | + case ARM_CPUID_PXA270_C5: | |
43 | + set_feature(env, ARM_FEATURE_XSCALE); | |
44 | + /* JTAG_ID is ((id << 28) | 0x09265013) */ | |
45 | + env->cp15.c0_cachetype = 0xd172172; | |
25 | 46 | break; |
26 | 47 | default: |
27 | 48 | cpu_abort(env, "Bad CPU ID: %x\n", id); |
... | ... | @@ -68,6 +89,18 @@ struct arm_cpu_t { |
68 | 89 | static const struct arm_cpu_t arm_cpu_names[] = { |
69 | 90 | { ARM_CPUID_ARM926, "arm926"}, |
70 | 91 | { ARM_CPUID_ARM1026, "arm1026"}, |
92 | + { ARM_CPUID_PXA250, "pxa250" }, | |
93 | + { ARM_CPUID_PXA255, "pxa255" }, | |
94 | + { ARM_CPUID_PXA260, "pxa260" }, | |
95 | + { ARM_CPUID_PXA261, "pxa261" }, | |
96 | + { ARM_CPUID_PXA262, "pxa262" }, | |
97 | + { ARM_CPUID_PXA270, "pxa270" }, | |
98 | + { ARM_CPUID_PXA270_A0, "pxa270-a0" }, | |
99 | + { ARM_CPUID_PXA270_A1, "pxa270-a1" }, | |
100 | + { ARM_CPUID_PXA270_B0, "pxa270-b0" }, | |
101 | + { ARM_CPUID_PXA270_B1, "pxa270-b1" }, | |
102 | + { ARM_CPUID_PXA270_C0, "pxa270-c0" }, | |
103 | + { ARM_CPUID_PXA270_C5, "pxa270-c5" }, | |
71 | 104 | { 0, NULL} |
72 | 105 | }; |
73 | 106 | |
... | ... | @@ -132,6 +165,20 @@ target_phys_addr_t cpu_get_phys_page_debug(CPUState *env, target_ulong addr) |
132 | 165 | } |
133 | 166 | |
134 | 167 | /* These should probably raise undefined insn exceptions. */ |
168 | +void helper_set_cp(CPUState *env, uint32_t insn, uint32_t val) | |
169 | +{ | |
170 | + int op1 = (insn >> 8) & 0xf; | |
171 | + cpu_abort(env, "cp%i insn %08x\n", op1, insn); | |
172 | + return; | |
173 | +} | |
174 | + | |
175 | +uint32_t helper_get_cp(CPUState *env, uint32_t insn) | |
176 | +{ | |
177 | + int op1 = (insn >> 8) & 0xf; | |
178 | + cpu_abort(env, "cp%i insn %08x\n", op1, insn); | |
179 | + return 0; | |
180 | +} | |
181 | + | |
135 | 182 | void helper_set_cp15(CPUState *env, uint32_t insn, uint32_t val) |
136 | 183 | { |
137 | 184 | cpu_abort(env, "cp15 insn %08x\n", insn); |
... | ... | @@ -393,12 +440,16 @@ static int get_phys_addr(CPUState *env, uint32_t address, int access_type, |
393 | 440 | ap = (desc >> (4 + ((address >> 13) & 6))) & 3; |
394 | 441 | break; |
395 | 442 | case 3: /* 1k page. */ |
396 | - if (type == 1) { | |
397 | - /* Page translation fault. */ | |
398 | - code = 7; | |
399 | - goto do_fault; | |
443 | + if (arm_feature(env, ARM_FEATURE_XSCALE)) | |
444 | + phys_addr = (desc & 0xfffff000) | (address & 0xfff); | |
445 | + else { | |
446 | + if (type == 1) { | |
447 | + /* Page translation fault. */ | |
448 | + code = 7; | |
449 | + goto do_fault; | |
450 | + } | |
451 | + phys_addr = (desc & 0xfffffc00) | (address & 0x3ff); | |
400 | 452 | } |
401 | - phys_addr = (desc & 0xfffffc00) | (address & 0x3ff); | |
402 | 453 | ap = (desc >> 4) & 3; |
403 | 454 | break; |
404 | 455 | default: |
... | ... | @@ -461,6 +512,31 @@ target_phys_addr_t cpu_get_phys_page_debug(CPUState *env, target_ulong addr) |
461 | 512 | return phys_addr; |
462 | 513 | } |
463 | 514 | |
515 | +void helper_set_cp(CPUState *env, uint32_t insn, uint32_t val) | |
516 | +{ | |
517 | + int cp_num = (insn >> 8) & 0xf; | |
518 | + int cp_info = (insn >> 5) & 7; | |
519 | + int src = (insn >> 16) & 0xf; | |
520 | + int operand = insn & 0xf; | |
521 | + | |
522 | + if (env->cp[cp_num].cp_write) | |
523 | + env->cp[cp_num].cp_write(env->cp[cp_num].opaque, | |
524 | + cp_info, src, operand, val); | |
525 | +} | |
526 | + | |
527 | +uint32_t helper_get_cp(CPUState *env, uint32_t insn) | |
528 | +{ | |
529 | + int cp_num = (insn >> 8) & 0xf; | |
530 | + int cp_info = (insn >> 5) & 7; | |
531 | + int dest = (insn >> 16) & 0xf; | |
532 | + int operand = insn & 0xf; | |
533 | + | |
534 | + if (env->cp[cp_num].cp_read) | |
535 | + return env->cp[cp_num].cp_read(env->cp[cp_num].opaque, | |
536 | + cp_info, dest, operand); | |
537 | + return 0; | |
538 | +} | |
539 | + | |
464 | 540 | void helper_set_cp15(CPUState *env, uint32_t insn, uint32_t val) |
465 | 541 | { |
466 | 542 | uint32_t op2; |
... | ... | @@ -472,15 +548,23 @@ void helper_set_cp15(CPUState *env, uint32_t insn, uint32_t val) |
472 | 548 | case 1: /* System configuration. */ |
473 | 549 | switch (op2) { |
474 | 550 | case 0: |
475 | - env->cp15.c1_sys = val; | |
551 | + if (!arm_feature(env, ARM_FEATURE_XSCALE) || (insn & 0xf) == 0) | |
552 | + env->cp15.c1_sys = val; | |
476 | 553 | /* ??? Lots of these bits are not implemented. */ |
477 | 554 | /* This may enable/disable the MMU, so do a TLB flush. */ |
478 | 555 | tlb_flush(env, 1); |
479 | 556 | break; |
557 | + case 1: | |
558 | + /* XScale doesn't implement AUX CR (P-Bit) but allows | |
559 | + * writing with zero and reading. */ | |
560 | + if (arm_feature(env, ARM_FEATURE_XSCALE)) | |
561 | + break; | |
562 | + goto bad_reg; | |
480 | 563 | case 2: |
481 | 564 | env->cp15.c1_coproc = val; |
482 | 565 | /* ??? Is this safe when called from within a TB? */ |
483 | 566 | tb_flush(env); |
567 | + break; | |
484 | 568 | default: |
485 | 569 | goto bad_reg; |
486 | 570 | } |
... | ... | @@ -584,13 +668,21 @@ void helper_set_cp15(CPUState *env, uint32_t insn, uint32_t val) |
584 | 668 | case 14: /* Reserved. */ |
585 | 669 | goto bad_reg; |
586 | 670 | case 15: /* Implementation specific. */ |
587 | - /* ??? Internal registers not implemented. */ | |
671 | + if (arm_feature(env, ARM_FEATURE_XSCALE)) { | |
672 | + if (op2 == 0 && (insn & 0xf) == 1) { | |
673 | + /* Changes cp0 to cp13 behavior, so needs a TB flush. */ | |
674 | + tb_flush(env); | |
675 | + env->cp15.c15_cpar = (val & 0x3fff) | 2; | |
676 | + break; | |
677 | + } | |
678 | + goto bad_reg; | |
679 | + } | |
588 | 680 | break; |
589 | 681 | } |
590 | 682 | return; |
591 | 683 | bad_reg: |
592 | 684 | /* ??? For debugging only. Should raise illegal instruction exception. */ |
593 | - cpu_abort(env, "Unimplemented cp15 register read\n"); | |
685 | + cpu_abort(env, "Unimplemented cp15 register write\n"); | |
594 | 686 | } |
595 | 687 | |
596 | 688 | uint32_t helper_get_cp15(CPUState *env, uint32_t insn) |
... | ... | @@ -604,7 +696,7 @@ uint32_t helper_get_cp15(CPUState *env, uint32_t insn) |
604 | 696 | default: /* Device ID. */ |
605 | 697 | return env->cp15.c0_cpuid; |
606 | 698 | case 1: /* Cache Type. */ |
607 | - return 0x1dd20d2; | |
699 | + return env->cp15.c0_cachetype; | |
608 | 700 | case 2: /* TCM status. */ |
609 | 701 | return 0; |
610 | 702 | } |
... | ... | @@ -615,6 +707,8 @@ uint32_t helper_get_cp15(CPUState *env, uint32_t insn) |
615 | 707 | case 1: /* Auxiliary control register. */ |
616 | 708 | if (arm_feature(env, ARM_FEATURE_AUXCR)) |
617 | 709 | return 1; |
710 | + if (arm_feature(env, ARM_FEATURE_XSCALE)) | |
711 | + return 0; | |
618 | 712 | goto bad_reg; |
619 | 713 | case 2: /* Coprocessor access register. */ |
620 | 714 | return env->cp15.c1_coproc; |
... | ... | @@ -649,7 +743,7 @@ uint32_t helper_get_cp15(CPUState *env, uint32_t insn) |
649 | 743 | } |
650 | 744 | case 7: /* Cache control. */ |
651 | 745 | /* ??? This is for test, clean and invaidate operations that set the |
652 | - Z flag. We can't represent N = Z = 1, so it also clears clears | |
746 | + Z flag. We can't represent N = Z = 1, so it also clears | |
653 | 747 | the N flag. Oh well. */ |
654 | 748 | env->NZF = 0; |
655 | 749 | return 0; |
... | ... | @@ -682,7 +776,12 @@ uint32_t helper_get_cp15(CPUState *env, uint32_t insn) |
682 | 776 | case 14: /* Reserved. */ |
683 | 777 | goto bad_reg; |
684 | 778 | case 15: /* Implementation specific. */ |
685 | - /* ??? Internal registers not implemented. */ | |
779 | + if (arm_feature(env, ARM_FEATURE_XSCALE)) { | |
780 | + if (op2 == 0 && (insn & 0xf) == 1) | |
781 | + return env->cp15.c15_cpar; | |
782 | + | |
783 | + goto bad_reg; | |
784 | + } | |
686 | 785 | return 0; |
687 | 786 | } |
688 | 787 | bad_reg: |
... | ... | @@ -691,4 +790,18 @@ bad_reg: |
691 | 790 | return 0; |
692 | 791 | } |
693 | 792 | |
793 | +void cpu_arm_set_cp_io(CPUARMState *env, int cpnum, | |
794 | + ARMReadCPFunc *cp_read, ARMWriteCPFunc *cp_write, | |
795 | + void *opaque) | |
796 | +{ | |
797 | + if (cpnum < 0 || cpnum > 14) { | |
798 | + cpu_abort(env, "Bad coprocessor number: %i\n", cpnum); | |
799 | + return; | |
800 | + } | |
801 | + | |
802 | + env->cp[cpnum].cp_read = cp_read; | |
803 | + env->cp[cpnum].cp_write = cp_write; | |
804 | + env->cp[cpnum].opaque = opaque; | |
805 | +} | |
806 | + | |
694 | 807 | #endif | ... | ... |
target-arm/op.c
... | ... | @@ -1142,12 +1142,24 @@ void OPPROTO op_vfp_mdrr(void) |
1142 | 1142 | FT0d = u.d; |
1143 | 1143 | } |
1144 | 1144 | |
1145 | -/* Copy the most significant bit to T0 to all bits of T1. */ | |
1145 | +/* Copy the most significant bit of T0 to all bits of T1. */ | |
1146 | 1146 | void OPPROTO op_signbit_T1_T0(void) |
1147 | 1147 | { |
1148 | 1148 | T1 = (int32_t)T0 >> 31; |
1149 | 1149 | } |
1150 | 1150 | |
1151 | +void OPPROTO op_movl_cp_T0(void) | |
1152 | +{ | |
1153 | + helper_set_cp(env, PARAM1, T0); | |
1154 | + FORCE_RET(); | |
1155 | +} | |
1156 | + | |
1157 | +void OPPROTO op_movl_T0_cp(void) | |
1158 | +{ | |
1159 | + T0 = helper_get_cp(env, PARAM1); | |
1160 | + FORCE_RET(); | |
1161 | +} | |
1162 | + | |
1151 | 1163 | void OPPROTO op_movl_cp15_T0(void) |
1152 | 1164 | { |
1153 | 1165 | helper_set_cp15(env, PARAM1, T0); | ... | ... |
target-arm/translate.c
... | ... | @@ -492,6 +492,34 @@ static inline void gen_mov_vreg_F0(int dp, int reg) |
492 | 492 | gen_op_vfp_setreg_F0s(vfp_reg_offset(dp, reg)); |
493 | 493 | } |
494 | 494 | |
495 | +/* Disassemble system coprocessor instruction. Return nonzero if | |
496 | + instruction is not defined. */ | |
497 | +static int disas_cp_insn(CPUState *env, DisasContext *s, uint32_t insn) | |
498 | +{ | |
499 | + uint32_t rd = (insn >> 12) & 0xf; | |
500 | + uint32_t cp = (insn >> 8) & 0xf; | |
501 | + if (IS_USER(s)) { | |
502 | + return 1; | |
503 | + } | |
504 | + | |
505 | + if (insn & (1 << 20)) { | |
506 | + if (!env->cp[cp].cp_read) | |
507 | + return 1; | |
508 | + gen_op_movl_T0_im((uint32_t) s->pc); | |
509 | + gen_op_movl_reg_TN[0][15](); | |
510 | + gen_op_movl_T0_cp(insn); | |
511 | + gen_movl_reg_T0(s, rd); | |
512 | + } else { | |
513 | + if (!env->cp[cp].cp_write) | |
514 | + return 1; | |
515 | + gen_op_movl_T0_im((uint32_t) s->pc); | |
516 | + gen_op_movl_reg_TN[0][15](); | |
517 | + gen_movl_T0_reg(s, rd); | |
518 | + gen_op_movl_cp_T0(insn); | |
519 | + } | |
520 | + return 0; | |
521 | +} | |
522 | + | |
495 | 523 | /* Disassemble system coprocessor (cp15) instruction. Return nonzero if |
496 | 524 | instruction is not defined. */ |
497 | 525 | static int disas_cp15_insn(DisasContext *s, uint32_t insn) |
... | ... | @@ -1812,7 +1840,16 @@ static void disas_arm_insn(CPUState * env, DisasContext *s) |
1812 | 1840 | case 0xe: |
1813 | 1841 | /* Coprocessor. */ |
1814 | 1842 | op1 = (insn >> 8) & 0xf; |
1843 | + if (arm_feature(env, ARM_FEATURE_XSCALE) && | |
1844 | + ((env->cp15.c15_cpar ^ 0x3fff) & (1 << op1))) | |
1845 | + goto illegal_op; | |
1815 | 1846 | switch (op1) { |
1847 | + case 0 ... 1: | |
1848 | + case 2 ... 9: | |
1849 | + case 12 ... 14: | |
1850 | + if (disas_cp_insn (env, s, insn)) | |
1851 | + goto illegal_op; | |
1852 | + break; | |
1816 | 1853 | case 10: |
1817 | 1854 | case 11: |
1818 | 1855 | if (disas_vfp_insn (env, s, insn)) | ... | ... |
vl.h