2 * ASPEED Interrupt Controller (New)
4 * Andrew Jeffery <andrew@aj.id.au>
6 * Copyright 2015, 2016 IBM Corp.
8 * This code is licensed under the GPL version 2 or later. See
9 * the COPYING file in the top-level directory.
12 /* The hardware exposes two register sets, a legacy set and a 'new' set. The
13 * model implements the 'new' register set, and logs warnings on accesses to
14 * the legacy IO space.
16 * The hardware uses 32bit registers to manage 51 IRQs, with low and high
17 * registers for each conceptual register. The device model's implementation
18 * uses 64bit data types to store both low and high register values (in the one
19 * member), but must cope with access offset values in multiples of 4 passed to
20 * the callbacks. As such the read() and write() implementations process the
21 * provided offset to understand whether the access is requesting the lower or
22 * upper 32 bits of the 64bit member.
24 * Additionally, the "Interrupt Enable", "Edge Status" and "Software Interrupt"
25 * fields have separate "enable"/"status" and "clear" registers, where set bits
26 * are written to one or the other to change state (avoiding a
27 * read-modify-write sequence).
30 #include "qemu/osdep.h"
31 #include "hw/intc/aspeed_vic.h"
33 #include "migration/vmstate.h"
34 #include "qemu/bitops.h"
36 #include "qemu/module.h"
39 #define AVIC_NEW_BASE_OFFSET 0x80
41 #define AVIC_L_MASK 0xFFFFFFFFU
42 #define AVIC_H_MASK 0x0007FFFFU
43 #define AVIC_EVENT_W_MASK (0x78000ULL << 32)
45 static void aspeed_vic_update(AspeedVICState *s)
47 uint64_t new = (s->raw & s->enable);
50 flags = new & s->select;
51 trace_aspeed_vic_update_fiq(!!flags);
52 qemu_set_irq(s->fiq, !!flags);
54 flags = new & ~s->select;
55 trace_aspeed_vic_update_irq(!!flags);
56 qemu_set_irq(s->irq, !!flags);
59 static void aspeed_vic_set_irq(void *opaque, int irq, int level)
63 AspeedVICState *s = (AspeedVICState *)opaque;
65 if (irq > ASPEED_VIC_NR_IRQS) {
66 qemu_log_mask(LOG_GUEST_ERROR, "%s: Invalid interrupt number: %d\n",
71 trace_aspeed_vic_set_irq(irq, level);
74 if (s->sense & irq_mask) {
76 if (s->event & irq_mask) {
83 s->raw = deposit64(s->raw, irq, 1, raise);
85 uint64_t old_level = s->level & irq_mask;
88 if (s->dual_edge & irq_mask) {
89 raise = (!!old_level) != (!!level);
91 if (s->event & irq_mask) {
92 /* rising-sensitive */
93 raise = !old_level && level;
95 /* falling-sensitive */
96 raise = old_level && !level;
100 s->raw = deposit64(s->raw, irq, 1, raise);
103 s->level = deposit64(s->level, irq, 1, level);
104 aspeed_vic_update(s);
107 static uint64_t aspeed_vic_read(void *opaque, hwaddr offset, unsigned size)
109 AspeedVICState *s = (AspeedVICState *)opaque;
114 if (offset < AVIC_NEW_BASE_OFFSET) {
118 high = !!(offset & 0x4);
119 n_offset = (offset & ~0x4);
123 case 0x80: /* IRQ Status */
125 val = s->raw & ~s->select & s->enable;
127 case 0x88: /* FIQ Status */
129 val = s->raw & s->select & s->enable;
131 case 0x90: /* Raw Interrupt Status */
135 case 0x98: /* Interrupt Selection */
139 case 0xa0: /* Interrupt Enable */
143 case 0xb0: /* Software Interrupt */
147 case 0xc0: /* Interrupt Sensitivity */
151 case 0xc8: /* Interrupt Both Edge Trigger Control */
155 case 0xd0: /* Interrupt Event */
159 case 0xe0: /* Edge Triggered Interrupt Status */
160 val = s->raw & ~s->sense;
163 case 0xa8: /* Interrupt Enable Clear */
164 case 0xb8: /* Software Interrupt Clear */
165 case 0xd8: /* Edge Triggered Interrupt Clear */
166 qemu_log_mask(LOG_GUEST_ERROR,
167 "%s: Read of write-only register with offset 0x%"
168 HWADDR_PRIx "\n", __func__, offset);
172 qemu_log_mask(LOG_GUEST_ERROR,
173 "%s: Bad register at offset 0x%" HWADDR_PRIx "\n",
179 val = extract64(val, 32, 19);
181 val = extract64(val, 0, 32);
183 trace_aspeed_vic_read(offset, size, val);
187 static void aspeed_vic_write(void *opaque, hwaddr offset, uint64_t data,
190 AspeedVICState *s = (AspeedVICState *)opaque;
194 if (offset < AVIC_NEW_BASE_OFFSET) {
198 high = !!(offset & 0x4);
199 n_offset = (offset & ~0x4);
202 trace_aspeed_vic_write(offset, size, data);
204 /* Given we have members using separate enable/clear registers, deposit64()
205 * isn't quite the tool for the job. Instead, relocate the incoming bits to
206 * the required bit offset based on the provided access address
216 case 0x98: /* Interrupt Selection */
218 /* Register has deposit64() semantics - overwrite requested 32 bits */
220 s->select &= AVIC_L_MASK;
222 s->select &= ((uint64_t) AVIC_H_MASK) << 32;
226 case 0xa0: /* Interrupt Enable */
230 case 0xa8: /* Interrupt Enable Clear */
234 case 0xb0: /* Software Interrupt */
236 qemu_log_mask(LOG_UNIMP, "%s: Software interrupts unavailable. "
237 "IRQs requested: 0x%016" PRIx64 "\n", __func__, data);
239 case 0xb8: /* Software Interrupt Clear */
241 qemu_log_mask(LOG_UNIMP, "%s: Software interrupts unavailable. "
242 "IRQs to be cleared: 0x%016" PRIx64 "\n", __func__, data);
244 case 0xd0: /* Interrupt Event */
245 /* Register has deposit64() semantics - overwrite the top four valid
246 * IRQ bits, as only the top four IRQs (GPIOs) can change their event
249 s->event &= ~AVIC_EVENT_W_MASK;
250 s->event |= (data & AVIC_EVENT_W_MASK);
252 qemu_log_mask(LOG_GUEST_ERROR,
253 "Ignoring invalid write to interrupt event register");
256 case 0xd8: /* Edge Triggered Interrupt Clear */
258 s->raw &= ~(data & ~s->sense);
260 case 0x80: /* IRQ Status */
262 case 0x88: /* FIQ Status */
264 case 0x90: /* Raw Interrupt Status */
266 case 0xc0: /* Interrupt Sensitivity */
268 case 0xc8: /* Interrupt Both Edge Trigger Control */
270 case 0xe0: /* Edge Triggered Interrupt Status */
271 qemu_log_mask(LOG_GUEST_ERROR,
272 "%s: Write of read-only register with offset 0x%"
273 HWADDR_PRIx "\n", __func__, offset);
277 qemu_log_mask(LOG_GUEST_ERROR,
278 "%s: Bad register at offset 0x%" HWADDR_PRIx "\n",
282 aspeed_vic_update(s);
285 static const MemoryRegionOps aspeed_vic_ops = {
286 .read = aspeed_vic_read,
287 .write = aspeed_vic_write,
288 .endianness = DEVICE_LITTLE_ENDIAN,
289 .valid.min_access_size = 4,
290 .valid.max_access_size = 4,
291 .valid.unaligned = false,
294 static void aspeed_vic_reset(DeviceState *dev)
296 AspeedVICState *s = ASPEED_VIC(dev);
303 s->sense = 0x1F07FFF8FFFFULL;
304 s->dual_edge = 0xF800070000ULL;
305 s->event = 0x5F07FFF8FFFFULL;
308 #define AVIC_IO_REGION_SIZE 0x20000
310 static void aspeed_vic_realize(DeviceState *dev, Error **errp)
312 SysBusDevice *sbd = SYS_BUS_DEVICE(dev);
313 AspeedVICState *s = ASPEED_VIC(dev);
315 memory_region_init_io(&s->iomem, OBJECT(s), &aspeed_vic_ops, s,
316 TYPE_ASPEED_VIC, AVIC_IO_REGION_SIZE);
318 sysbus_init_mmio(sbd, &s->iomem);
320 qdev_init_gpio_in(dev, aspeed_vic_set_irq, ASPEED_VIC_NR_IRQS);
321 sysbus_init_irq(sbd, &s->irq);
322 sysbus_init_irq(sbd, &s->fiq);
325 static const VMStateDescription vmstate_aspeed_vic = {
326 .name = "aspeed.new-vic",
328 .minimum_version_id = 1,
329 .fields = (VMStateField[]) {
330 VMSTATE_UINT64(level, AspeedVICState),
331 VMSTATE_UINT64(raw, AspeedVICState),
332 VMSTATE_UINT64(select, AspeedVICState),
333 VMSTATE_UINT64(enable, AspeedVICState),
334 VMSTATE_UINT64(trigger, AspeedVICState),
335 VMSTATE_UINT64(sense, AspeedVICState),
336 VMSTATE_UINT64(dual_edge, AspeedVICState),
337 VMSTATE_UINT64(event, AspeedVICState),
338 VMSTATE_END_OF_LIST()
342 static void aspeed_vic_class_init(ObjectClass *klass, void *data)
344 DeviceClass *dc = DEVICE_CLASS(klass);
345 dc->realize = aspeed_vic_realize;
346 dc->reset = aspeed_vic_reset;
347 dc->desc = "ASPEED Interrupt Controller (New)";
348 dc->vmsd = &vmstate_aspeed_vic;
351 static const TypeInfo aspeed_vic_info = {
352 .name = TYPE_ASPEED_VIC,
353 .parent = TYPE_SYS_BUS_DEVICE,
354 .instance_size = sizeof(AspeedVICState),
355 .class_init = aspeed_vic_class_init,
358 static void aspeed_vic_register_types(void)
360 type_register_static(&aspeed_vic_info);
363 type_init(aspeed_vic_register_types);