root / hw / pl190.c @ 5fafdf24
History | View | Annotate | Download (6.4 kB)
1 | 5fafdf24 | ths | /*
|
---|---|---|---|
2 | cdbdb648 | pbrook | * Arm PrimeCell PL190 Vector Interrupt Controller
|
3 | cdbdb648 | pbrook | *
|
4 | cdbdb648 | pbrook | * Copyright (c) 2006 CodeSourcery.
|
5 | cdbdb648 | pbrook | * Written by Paul Brook
|
6 | cdbdb648 | pbrook | *
|
7 | cdbdb648 | pbrook | * This code is licenced under the GPL.
|
8 | cdbdb648 | pbrook | */
|
9 | cdbdb648 | pbrook | |
10 | cdbdb648 | pbrook | #include "vl.h" |
11 | cdbdb648 | pbrook | #include "arm_pic.h" |
12 | cdbdb648 | pbrook | |
13 | cdbdb648 | pbrook | /* The number of virtual priority levels. 16 user vectors plus the
|
14 | cdbdb648 | pbrook | unvectored IRQ. Chained interrupts would require an additional level
|
15 | cdbdb648 | pbrook | if implemented. */
|
16 | cdbdb648 | pbrook | |
17 | cdbdb648 | pbrook | #define PL190_NUM_PRIO 17 |
18 | cdbdb648 | pbrook | |
19 | cdbdb648 | pbrook | typedef struct { |
20 | cdbdb648 | pbrook | uint32_t base; |
21 | cdbdb648 | pbrook | DisplayState *ds; |
22 | cdbdb648 | pbrook | uint32_t level; |
23 | cdbdb648 | pbrook | uint32_t soft_level; |
24 | cdbdb648 | pbrook | uint32_t irq_enable; |
25 | cdbdb648 | pbrook | uint32_t fiq_select; |
26 | cdbdb648 | pbrook | uint32_t default_addr; |
27 | cdbdb648 | pbrook | uint8_t vect_control[16];
|
28 | cdbdb648 | pbrook | uint32_t vect_addr[PL190_NUM_PRIO]; |
29 | cdbdb648 | pbrook | /* Mask containing interrupts with higher priority than this one. */
|
30 | cdbdb648 | pbrook | uint32_t prio_mask[PL190_NUM_PRIO + 1];
|
31 | cdbdb648 | pbrook | int protected;
|
32 | cdbdb648 | pbrook | /* Current priority level. */
|
33 | cdbdb648 | pbrook | int priority;
|
34 | cdbdb648 | pbrook | int prev_prio[PL190_NUM_PRIO];
|
35 | d537cf6c | pbrook | qemu_irq irq; |
36 | d537cf6c | pbrook | qemu_irq fiq; |
37 | cdbdb648 | pbrook | } pl190_state; |
38 | cdbdb648 | pbrook | |
39 | cdbdb648 | pbrook | static const unsigned char pl190_id[] = |
40 | cdbdb648 | pbrook | { 0x90, 0x11, 0x04, 0x00, 0x0D, 0xf0, 0x05, 0xb1 }; |
41 | cdbdb648 | pbrook | |
42 | cdbdb648 | pbrook | static inline uint32_t pl190_irq_level(pl190_state *s) |
43 | cdbdb648 | pbrook | { |
44 | cdbdb648 | pbrook | return (s->level | s->soft_level) & s->irq_enable & ~s->fiq_select;
|
45 | cdbdb648 | pbrook | } |
46 | cdbdb648 | pbrook | |
47 | cdbdb648 | pbrook | /* Update interrupts. */
|
48 | cdbdb648 | pbrook | static void pl190_update(pl190_state *s) |
49 | cdbdb648 | pbrook | { |
50 | cdbdb648 | pbrook | uint32_t level = pl190_irq_level(s); |
51 | cdbdb648 | pbrook | int set;
|
52 | cdbdb648 | pbrook | |
53 | cdbdb648 | pbrook | set = (level & s->prio_mask[s->priority]) != 0;
|
54 | d537cf6c | pbrook | qemu_set_irq(s->irq, set); |
55 | cdbdb648 | pbrook | set = ((s->level | s->soft_level) & s->fiq_select) != 0;
|
56 | d537cf6c | pbrook | qemu_set_irq(s->fiq, set); |
57 | cdbdb648 | pbrook | } |
58 | cdbdb648 | pbrook | |
59 | cdbdb648 | pbrook | static void pl190_set_irq(void *opaque, int irq, int level) |
60 | cdbdb648 | pbrook | { |
61 | cdbdb648 | pbrook | pl190_state *s = (pl190_state *)opaque; |
62 | cdbdb648 | pbrook | |
63 | cdbdb648 | pbrook | if (level)
|
64 | cdbdb648 | pbrook | s->level |= 1u << irq;
|
65 | cdbdb648 | pbrook | else
|
66 | cdbdb648 | pbrook | s->level &= ~(1u << irq);
|
67 | cdbdb648 | pbrook | pl190_update(s); |
68 | cdbdb648 | pbrook | } |
69 | cdbdb648 | pbrook | |
70 | cdbdb648 | pbrook | static void pl190_update_vectors(pl190_state *s) |
71 | cdbdb648 | pbrook | { |
72 | cdbdb648 | pbrook | uint32_t mask; |
73 | cdbdb648 | pbrook | int i;
|
74 | cdbdb648 | pbrook | int n;
|
75 | cdbdb648 | pbrook | |
76 | cdbdb648 | pbrook | mask = 0;
|
77 | cdbdb648 | pbrook | for (i = 0; i < 16; i++) |
78 | cdbdb648 | pbrook | { |
79 | cdbdb648 | pbrook | s->prio_mask[i] = mask; |
80 | cdbdb648 | pbrook | if (s->vect_control[i] & 0x20) |
81 | cdbdb648 | pbrook | { |
82 | cdbdb648 | pbrook | n = s->vect_control[i] & 0x1f;
|
83 | cdbdb648 | pbrook | mask |= 1 << n;
|
84 | cdbdb648 | pbrook | } |
85 | cdbdb648 | pbrook | } |
86 | cdbdb648 | pbrook | s->prio_mask[16] = mask;
|
87 | cdbdb648 | pbrook | pl190_update(s); |
88 | cdbdb648 | pbrook | } |
89 | cdbdb648 | pbrook | |
90 | cdbdb648 | pbrook | static uint32_t pl190_read(void *opaque, target_phys_addr_t offset) |
91 | cdbdb648 | pbrook | { |
92 | cdbdb648 | pbrook | pl190_state *s = (pl190_state *)opaque; |
93 | cdbdb648 | pbrook | int i;
|
94 | cdbdb648 | pbrook | |
95 | cdbdb648 | pbrook | offset -= s->base; |
96 | cdbdb648 | pbrook | if (offset >= 0xfe0 && offset < 0x1000) { |
97 | cdbdb648 | pbrook | return pl190_id[(offset - 0xfe0) >> 2]; |
98 | cdbdb648 | pbrook | } |
99 | cdbdb648 | pbrook | if (offset >= 0x100 && offset < 0x140) { |
100 | cdbdb648 | pbrook | return s->vect_addr[(offset - 0x100) >> 2]; |
101 | cdbdb648 | pbrook | } |
102 | cdbdb648 | pbrook | if (offset >= 0x200 && offset < 0x240) { |
103 | cdbdb648 | pbrook | return s->vect_control[(offset - 0x200) >> 2]; |
104 | cdbdb648 | pbrook | } |
105 | cdbdb648 | pbrook | switch (offset >> 2) { |
106 | cdbdb648 | pbrook | case 0: /* IRQSTATUS */ |
107 | cdbdb648 | pbrook | return pl190_irq_level(s);
|
108 | cdbdb648 | pbrook | case 1: /* FIQSATUS */ |
109 | cdbdb648 | pbrook | return (s->level | s->soft_level) & s->fiq_select;
|
110 | cdbdb648 | pbrook | case 2: /* RAWINTR */ |
111 | cdbdb648 | pbrook | return s->level | s->soft_level;
|
112 | cdbdb648 | pbrook | case 3: /* INTSELECT */ |
113 | cdbdb648 | pbrook | return s->fiq_select;
|
114 | cdbdb648 | pbrook | case 4: /* INTENABLE */ |
115 | cdbdb648 | pbrook | return s->irq_enable;
|
116 | cdbdb648 | pbrook | case 6: /* SOFTINT */ |
117 | cdbdb648 | pbrook | return s->soft_level;
|
118 | cdbdb648 | pbrook | case 8: /* PROTECTION */ |
119 | cdbdb648 | pbrook | return s->protected;
|
120 | cdbdb648 | pbrook | case 12: /* VECTADDR */ |
121 | cdbdb648 | pbrook | /* Read vector address at the start of an ISR. Increases the
|
122 | cdbdb648 | pbrook | current priority level to that of the current interrupt. */
|
123 | cdbdb648 | pbrook | for (i = 0; i < s->priority; i++) |
124 | cdbdb648 | pbrook | { |
125 | cdbdb648 | pbrook | if ((s->level | s->soft_level) & s->prio_mask[i])
|
126 | cdbdb648 | pbrook | break;
|
127 | cdbdb648 | pbrook | } |
128 | cdbdb648 | pbrook | /* Reading this value with no pending interrupts is undefined.
|
129 | cdbdb648 | pbrook | We return the default address. */
|
130 | cdbdb648 | pbrook | if (i == PL190_NUM_PRIO)
|
131 | cdbdb648 | pbrook | return s->vect_addr[16]; |
132 | cdbdb648 | pbrook | if (i < s->priority)
|
133 | cdbdb648 | pbrook | { |
134 | cdbdb648 | pbrook | s->prev_prio[i] = s->priority; |
135 | cdbdb648 | pbrook | s->priority = i; |
136 | cdbdb648 | pbrook | pl190_update(s); |
137 | cdbdb648 | pbrook | } |
138 | cdbdb648 | pbrook | return s->vect_addr[s->priority];
|
139 | cdbdb648 | pbrook | case 13: /* DEFVECTADDR */ |
140 | cdbdb648 | pbrook | return s->vect_addr[16]; |
141 | cdbdb648 | pbrook | default:
|
142 | cdbdb648 | pbrook | cpu_abort (cpu_single_env, "pl190_read: Bad offset %x\n", offset);
|
143 | cdbdb648 | pbrook | return 0; |
144 | cdbdb648 | pbrook | } |
145 | cdbdb648 | pbrook | } |
146 | cdbdb648 | pbrook | |
147 | cdbdb648 | pbrook | static void pl190_write(void *opaque, target_phys_addr_t offset, uint32_t val) |
148 | cdbdb648 | pbrook | { |
149 | cdbdb648 | pbrook | pl190_state *s = (pl190_state *)opaque; |
150 | cdbdb648 | pbrook | |
151 | cdbdb648 | pbrook | offset -= s->base; |
152 | cdbdb648 | pbrook | if (offset >= 0x100 && offset < 0x140) { |
153 | cdbdb648 | pbrook | s->vect_addr[(offset - 0x100) >> 2] = val; |
154 | cdbdb648 | pbrook | pl190_update_vectors(s); |
155 | cdbdb648 | pbrook | return;
|
156 | cdbdb648 | pbrook | } |
157 | cdbdb648 | pbrook | if (offset >= 0x200 && offset < 0x240) { |
158 | cdbdb648 | pbrook | s->vect_control[(offset - 0x200) >> 2] = val; |
159 | cdbdb648 | pbrook | pl190_update_vectors(s); |
160 | cdbdb648 | pbrook | return;
|
161 | cdbdb648 | pbrook | } |
162 | cdbdb648 | pbrook | switch (offset >> 2) { |
163 | cdbdb648 | pbrook | case 0: /* SELECT */ |
164 | cdbdb648 | pbrook | /* This is a readonly register, but linux tries to write to it
|
165 | cdbdb648 | pbrook | anyway. Ignore the write. */
|
166 | cdbdb648 | pbrook | break;
|
167 | cdbdb648 | pbrook | case 3: /* INTSELECT */ |
168 | cdbdb648 | pbrook | s->fiq_select = val; |
169 | cdbdb648 | pbrook | break;
|
170 | cdbdb648 | pbrook | case 4: /* INTENABLE */ |
171 | cdbdb648 | pbrook | s->irq_enable |= val; |
172 | cdbdb648 | pbrook | break;
|
173 | cdbdb648 | pbrook | case 5: /* INTENCLEAR */ |
174 | cdbdb648 | pbrook | s->irq_enable &= ~val; |
175 | cdbdb648 | pbrook | break;
|
176 | cdbdb648 | pbrook | case 6: /* SOFTINT */ |
177 | cdbdb648 | pbrook | s->soft_level |= val; |
178 | cdbdb648 | pbrook | break;
|
179 | cdbdb648 | pbrook | case 7: /* SOFTINTCLEAR */ |
180 | cdbdb648 | pbrook | s->soft_level &= ~val; |
181 | cdbdb648 | pbrook | break;
|
182 | cdbdb648 | pbrook | case 8: /* PROTECTION */ |
183 | cdbdb648 | pbrook | /* TODO: Protection (supervisor only access) is not implemented. */
|
184 | cdbdb648 | pbrook | s->protected = val & 1;
|
185 | cdbdb648 | pbrook | break;
|
186 | cdbdb648 | pbrook | case 12: /* VECTADDR */ |
187 | cdbdb648 | pbrook | /* Restore the previous priority level. The value written is
|
188 | cdbdb648 | pbrook | ignored. */
|
189 | cdbdb648 | pbrook | if (s->priority < PL190_NUM_PRIO)
|
190 | cdbdb648 | pbrook | s->priority = s->prev_prio[s->priority]; |
191 | cdbdb648 | pbrook | break;
|
192 | cdbdb648 | pbrook | case 13: /* DEFVECTADDR */ |
193 | cdbdb648 | pbrook | s->default_addr = val; |
194 | cdbdb648 | pbrook | break;
|
195 | cdbdb648 | pbrook | case 0xc0: /* ITCR */ |
196 | cdbdb648 | pbrook | if (val)
|
197 | cdbdb648 | pbrook | cpu_abort(cpu_single_env, "pl190: Test mode not implemented\n");
|
198 | cdbdb648 | pbrook | break;
|
199 | cdbdb648 | pbrook | default:
|
200 | cdbdb648 | pbrook | cpu_abort(cpu_single_env, "pl190_write: Bad offset %x\n", offset);
|
201 | cdbdb648 | pbrook | return;
|
202 | cdbdb648 | pbrook | } |
203 | cdbdb648 | pbrook | pl190_update(s); |
204 | cdbdb648 | pbrook | } |
205 | cdbdb648 | pbrook | |
206 | cdbdb648 | pbrook | static CPUReadMemoryFunc *pl190_readfn[] = {
|
207 | cdbdb648 | pbrook | pl190_read, |
208 | cdbdb648 | pbrook | pl190_read, |
209 | cdbdb648 | pbrook | pl190_read |
210 | cdbdb648 | pbrook | }; |
211 | cdbdb648 | pbrook | |
212 | cdbdb648 | pbrook | static CPUWriteMemoryFunc *pl190_writefn[] = {
|
213 | cdbdb648 | pbrook | pl190_write, |
214 | cdbdb648 | pbrook | pl190_write, |
215 | cdbdb648 | pbrook | pl190_write |
216 | cdbdb648 | pbrook | }; |
217 | cdbdb648 | pbrook | |
218 | cdbdb648 | pbrook | void pl190_reset(pl190_state *s)
|
219 | cdbdb648 | pbrook | { |
220 | cdbdb648 | pbrook | int i;
|
221 | cdbdb648 | pbrook | |
222 | cdbdb648 | pbrook | for (i = 0; i < 16; i++) |
223 | cdbdb648 | pbrook | { |
224 | cdbdb648 | pbrook | s->vect_addr[i] = 0;
|
225 | cdbdb648 | pbrook | s->vect_control[i] = 0;
|
226 | cdbdb648 | pbrook | } |
227 | cdbdb648 | pbrook | s->vect_addr[16] = 0; |
228 | cdbdb648 | pbrook | s->prio_mask[17] = 0xffffffff; |
229 | cdbdb648 | pbrook | s->priority = PL190_NUM_PRIO; |
230 | cdbdb648 | pbrook | pl190_update_vectors(s); |
231 | cdbdb648 | pbrook | } |
232 | cdbdb648 | pbrook | |
233 | d537cf6c | pbrook | qemu_irq *pl190_init(uint32_t base, qemu_irq irq, qemu_irq fiq) |
234 | cdbdb648 | pbrook | { |
235 | cdbdb648 | pbrook | pl190_state *s; |
236 | d537cf6c | pbrook | qemu_irq *qi; |
237 | cdbdb648 | pbrook | int iomemtype;
|
238 | cdbdb648 | pbrook | |
239 | cdbdb648 | pbrook | s = (pl190_state *)qemu_mallocz(sizeof(pl190_state));
|
240 | cdbdb648 | pbrook | iomemtype = cpu_register_io_memory(0, pl190_readfn,
|
241 | cdbdb648 | pbrook | pl190_writefn, s); |
242 | 187337f8 | pbrook | cpu_register_physical_memory(base, 0x00001000, iomemtype);
|
243 | 069dd10a | pbrook | qi = qemu_allocate_irqs(pl190_set_irq, s, 32);
|
244 | cdbdb648 | pbrook | s->base = base; |
245 | cdbdb648 | pbrook | s->irq = irq; |
246 | cdbdb648 | pbrook | s->fiq = fiq; |
247 | cdbdb648 | pbrook | pl190_reset(s); |
248 | cdbdb648 | pbrook | /* ??? Save/restore. */
|
249 | d537cf6c | pbrook | return qi;
|
250 | cdbdb648 | pbrook | } |