xref: /qemu/hw/intc/allwinner-a10-pic.c (revision c3931ee8b42def4089831b4d79e93c5b05667ff6)
1*c3931ee8Sliguang /*
2*c3931ee8Sliguang  * Allwinner A10 interrupt controller device emulation
3*c3931ee8Sliguang  *
4*c3931ee8Sliguang  * Copyright (C) 2013 Li Guang
5*c3931ee8Sliguang  * Written by Li Guang <lig.fnst@cn.fujitsu.com>
6*c3931ee8Sliguang  *
7*c3931ee8Sliguang  * This program is free software; you can redistribute it and/or modify it
8*c3931ee8Sliguang  * under the terms of the GNU General Public License as published by the
9*c3931ee8Sliguang  * Free Software Foundation; either version 2 of the License, or
10*c3931ee8Sliguang  * (at your option) any later version.
11*c3931ee8Sliguang  *
12*c3931ee8Sliguang  * This program is distributed in the hope that it will be useful, but WITHOUT
13*c3931ee8Sliguang  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
14*c3931ee8Sliguang  * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
15*c3931ee8Sliguang  * for more details.
16*c3931ee8Sliguang  */
17*c3931ee8Sliguang 
18*c3931ee8Sliguang #include "hw/sysbus.h"
19*c3931ee8Sliguang #include "hw/devices.h"
20*c3931ee8Sliguang #include "sysemu/sysemu.h"
21*c3931ee8Sliguang #include "hw/intc/allwinner-a10-pic.h"
22*c3931ee8Sliguang 
23*c3931ee8Sliguang static void aw_a10_pic_update(AwA10PICState *s)
24*c3931ee8Sliguang {
25*c3931ee8Sliguang     uint8_t i;
26*c3931ee8Sliguang     int irq = 0, fiq = 0;
27*c3931ee8Sliguang 
28*c3931ee8Sliguang     for (i = 0; i < AW_A10_PIC_REG_NUM; i++) {
29*c3931ee8Sliguang         irq |= s->irq_pending[i] & ~s->mask[i];
30*c3931ee8Sliguang         fiq |= s->select[i] & s->irq_pending[i] & ~s->mask[i];
31*c3931ee8Sliguang     }
32*c3931ee8Sliguang 
33*c3931ee8Sliguang     qemu_set_irq(s->parent_irq, !!irq);
34*c3931ee8Sliguang     qemu_set_irq(s->parent_fiq, !!fiq);
35*c3931ee8Sliguang }
36*c3931ee8Sliguang 
37*c3931ee8Sliguang static void aw_a10_pic_set_irq(void *opaque, int irq, int level)
38*c3931ee8Sliguang {
39*c3931ee8Sliguang     AwA10PICState *s = opaque;
40*c3931ee8Sliguang 
41*c3931ee8Sliguang     if (level) {
42*c3931ee8Sliguang         set_bit(irq % 32, (void *)&s->irq_pending[irq / 32]);
43*c3931ee8Sliguang     }
44*c3931ee8Sliguang     aw_a10_pic_update(s);
45*c3931ee8Sliguang }
46*c3931ee8Sliguang 
47*c3931ee8Sliguang static uint64_t aw_a10_pic_read(void *opaque, hwaddr offset, unsigned size)
48*c3931ee8Sliguang {
49*c3931ee8Sliguang     AwA10PICState *s = opaque;
50*c3931ee8Sliguang     uint8_t index = (offset & 0xc) / 4;
51*c3931ee8Sliguang 
52*c3931ee8Sliguang     switch (offset) {
53*c3931ee8Sliguang     case AW_A10_PIC_VECTOR:
54*c3931ee8Sliguang         return s->vector;
55*c3931ee8Sliguang     case AW_A10_PIC_BASE_ADDR:
56*c3931ee8Sliguang         return s->base_addr;
57*c3931ee8Sliguang     case AW_A10_PIC_PROTECT:
58*c3931ee8Sliguang         return s->protect;
59*c3931ee8Sliguang     case AW_A10_PIC_NMI:
60*c3931ee8Sliguang         return s->nmi;
61*c3931ee8Sliguang     case AW_A10_PIC_IRQ_PENDING ... AW_A10_PIC_IRQ_PENDING + 8:
62*c3931ee8Sliguang         return s->irq_pending[index];
63*c3931ee8Sliguang     case AW_A10_PIC_FIQ_PENDING ... AW_A10_PIC_FIQ_PENDING + 8:
64*c3931ee8Sliguang         return s->fiq_pending[index];
65*c3931ee8Sliguang     case AW_A10_PIC_SELECT ... AW_A10_PIC_SELECT + 8:
66*c3931ee8Sliguang         return s->select[index];
67*c3931ee8Sliguang     case AW_A10_PIC_ENABLE ... AW_A10_PIC_ENABLE + 8:
68*c3931ee8Sliguang         return s->enable[index];
69*c3931ee8Sliguang     case AW_A10_PIC_MASK ... AW_A10_PIC_MASK + 8:
70*c3931ee8Sliguang         return s->mask[index];
71*c3931ee8Sliguang     default:
72*c3931ee8Sliguang         qemu_log_mask(LOG_GUEST_ERROR,
73*c3931ee8Sliguang                       "%s: Bad offset 0x%x\n",  __func__, (int)offset);
74*c3931ee8Sliguang         break;
75*c3931ee8Sliguang     }
76*c3931ee8Sliguang 
77*c3931ee8Sliguang     return 0;
78*c3931ee8Sliguang }
79*c3931ee8Sliguang 
80*c3931ee8Sliguang static void aw_a10_pic_write(void *opaque, hwaddr offset, uint64_t value,
81*c3931ee8Sliguang                              unsigned size)
82*c3931ee8Sliguang {
83*c3931ee8Sliguang     AwA10PICState *s = opaque;
84*c3931ee8Sliguang     uint8_t index = (offset & 0xc) / 4;
85*c3931ee8Sliguang 
86*c3931ee8Sliguang     switch (offset) {
87*c3931ee8Sliguang     case AW_A10_PIC_VECTOR:
88*c3931ee8Sliguang         s->vector = value & ~0x3;
89*c3931ee8Sliguang         break;
90*c3931ee8Sliguang     case AW_A10_PIC_BASE_ADDR:
91*c3931ee8Sliguang         s->base_addr = value & ~0x3;
92*c3931ee8Sliguang     case AW_A10_PIC_PROTECT:
93*c3931ee8Sliguang         s->protect = value;
94*c3931ee8Sliguang         break;
95*c3931ee8Sliguang     case AW_A10_PIC_NMI:
96*c3931ee8Sliguang         s->nmi = value;
97*c3931ee8Sliguang         break;
98*c3931ee8Sliguang     case AW_A10_PIC_IRQ_PENDING ... AW_A10_PIC_IRQ_PENDING + 8:
99*c3931ee8Sliguang         s->irq_pending[index] &= ~value;
100*c3931ee8Sliguang         break;
101*c3931ee8Sliguang     case AW_A10_PIC_FIQ_PENDING ... AW_A10_PIC_FIQ_PENDING + 8:
102*c3931ee8Sliguang         s->fiq_pending[index] &= ~value;
103*c3931ee8Sliguang         break;
104*c3931ee8Sliguang     case AW_A10_PIC_SELECT ... AW_A10_PIC_SELECT + 8:
105*c3931ee8Sliguang         s->select[index] = value;
106*c3931ee8Sliguang         break;
107*c3931ee8Sliguang     case AW_A10_PIC_ENABLE ... AW_A10_PIC_ENABLE + 8:
108*c3931ee8Sliguang         s->enable[index] = value;
109*c3931ee8Sliguang         break;
110*c3931ee8Sliguang     case AW_A10_PIC_MASK ... AW_A10_PIC_MASK + 8:
111*c3931ee8Sliguang         s->mask[index] = value;
112*c3931ee8Sliguang         break;
113*c3931ee8Sliguang     default:
114*c3931ee8Sliguang         qemu_log_mask(LOG_GUEST_ERROR,
115*c3931ee8Sliguang                       "%s: Bad offset 0x%x\n",  __func__, (int)offset);
116*c3931ee8Sliguang         break;
117*c3931ee8Sliguang     }
118*c3931ee8Sliguang 
119*c3931ee8Sliguang     aw_a10_pic_update(s);
120*c3931ee8Sliguang }
121*c3931ee8Sliguang 
122*c3931ee8Sliguang static const MemoryRegionOps aw_a10_pic_ops = {
123*c3931ee8Sliguang     .read = aw_a10_pic_read,
124*c3931ee8Sliguang     .write = aw_a10_pic_write,
125*c3931ee8Sliguang     .endianness = DEVICE_NATIVE_ENDIAN,
126*c3931ee8Sliguang };
127*c3931ee8Sliguang 
128*c3931ee8Sliguang static const VMStateDescription vmstate_aw_a10_pic = {
129*c3931ee8Sliguang     .name = "a10.pic",
130*c3931ee8Sliguang     .version_id = 1,
131*c3931ee8Sliguang     .minimum_version_id = 1,
132*c3931ee8Sliguang     .minimum_version_id_old = 1,
133*c3931ee8Sliguang     .fields = (VMStateField[]) {
134*c3931ee8Sliguang         VMSTATE_UINT32(vector, AwA10PICState),
135*c3931ee8Sliguang         VMSTATE_UINT32(base_addr, AwA10PICState),
136*c3931ee8Sliguang         VMSTATE_UINT32(protect, AwA10PICState),
137*c3931ee8Sliguang         VMSTATE_UINT32(nmi, AwA10PICState),
138*c3931ee8Sliguang         VMSTATE_UINT32_ARRAY(irq_pending, AwA10PICState, AW_A10_PIC_REG_NUM),
139*c3931ee8Sliguang         VMSTATE_UINT32_ARRAY(fiq_pending, AwA10PICState, AW_A10_PIC_REG_NUM),
140*c3931ee8Sliguang         VMSTATE_UINT32_ARRAY(enable, AwA10PICState, AW_A10_PIC_REG_NUM),
141*c3931ee8Sliguang         VMSTATE_UINT32_ARRAY(select, AwA10PICState, AW_A10_PIC_REG_NUM),
142*c3931ee8Sliguang         VMSTATE_UINT32_ARRAY(mask, AwA10PICState, AW_A10_PIC_REG_NUM),
143*c3931ee8Sliguang         VMSTATE_END_OF_LIST()
144*c3931ee8Sliguang     }
145*c3931ee8Sliguang };
146*c3931ee8Sliguang 
147*c3931ee8Sliguang static void aw_a10_pic_init(Object *obj)
148*c3931ee8Sliguang {
149*c3931ee8Sliguang     AwA10PICState *s = AW_A10_PIC(obj);
150*c3931ee8Sliguang     SysBusDevice *dev = SYS_BUS_DEVICE(obj);
151*c3931ee8Sliguang 
152*c3931ee8Sliguang      qdev_init_gpio_in(DEVICE(dev), aw_a10_pic_set_irq, AW_A10_PIC_INT_NR);
153*c3931ee8Sliguang      sysbus_init_irq(dev, &s->parent_irq);
154*c3931ee8Sliguang      sysbus_init_irq(dev, &s->parent_fiq);
155*c3931ee8Sliguang      memory_region_init_io(&s->iomem, OBJECT(s), &aw_a10_pic_ops, s,
156*c3931ee8Sliguang                            TYPE_AW_A10_PIC, 0x400);
157*c3931ee8Sliguang      sysbus_init_mmio(dev, &s->iomem);
158*c3931ee8Sliguang }
159*c3931ee8Sliguang 
160*c3931ee8Sliguang static void aw_a10_pic_reset(DeviceState *d)
161*c3931ee8Sliguang {
162*c3931ee8Sliguang     AwA10PICState *s = AW_A10_PIC(d);
163*c3931ee8Sliguang     uint8_t i;
164*c3931ee8Sliguang 
165*c3931ee8Sliguang     s->base_addr = 0;
166*c3931ee8Sliguang     s->protect = 0;
167*c3931ee8Sliguang     s->nmi = 0;
168*c3931ee8Sliguang     s->vector = 0;
169*c3931ee8Sliguang     for (i = 0; i < AW_A10_PIC_REG_NUM; i++) {
170*c3931ee8Sliguang         s->irq_pending[i] = 0;
171*c3931ee8Sliguang         s->fiq_pending[i] = 0;
172*c3931ee8Sliguang         s->select[i] = 0;
173*c3931ee8Sliguang         s->enable[i] = 0;
174*c3931ee8Sliguang         s->mask[i] = 0;
175*c3931ee8Sliguang     }
176*c3931ee8Sliguang }
177*c3931ee8Sliguang 
178*c3931ee8Sliguang static void aw_a10_pic_class_init(ObjectClass *klass, void *data)
179*c3931ee8Sliguang {
180*c3931ee8Sliguang     DeviceClass *dc = DEVICE_CLASS(klass);
181*c3931ee8Sliguang 
182*c3931ee8Sliguang     dc->reset = aw_a10_pic_reset;
183*c3931ee8Sliguang     dc->desc = "allwinner a10 pic";
184*c3931ee8Sliguang     dc->vmsd = &vmstate_aw_a10_pic;
185*c3931ee8Sliguang  }
186*c3931ee8Sliguang 
187*c3931ee8Sliguang static const TypeInfo aw_a10_pic_info = {
188*c3931ee8Sliguang     .name = TYPE_AW_A10_PIC,
189*c3931ee8Sliguang     .parent = TYPE_SYS_BUS_DEVICE,
190*c3931ee8Sliguang     .instance_size = sizeof(AwA10PICState),
191*c3931ee8Sliguang     .instance_init = aw_a10_pic_init,
192*c3931ee8Sliguang     .class_init = aw_a10_pic_class_init,
193*c3931ee8Sliguang };
194*c3931ee8Sliguang 
195*c3931ee8Sliguang static void aw_a10_register_types(void)
196*c3931ee8Sliguang {
197*c3931ee8Sliguang     type_register_static(&aw_a10_pic_info);
198*c3931ee8Sliguang }
199*c3931ee8Sliguang 
200*c3931ee8Sliguang type_init(aw_a10_register_types);
201