@@ -34,6 +34,28 @@
* have the exception level features present.
* + QOM property "gic-spi-num" sets the number of GIC Shared Peripheral
* Interrupts.
+ * QEMU interface forwarded from the GIC:
+ * + unnamed GPIO inputs: (where P is number of GIC SPIs, i.e. num-irq - 32)
+ * [0..P-1] GIC SPIs
+ * [P..P+31] PPIs for CPU 0
+ * [P+32..P+63] PPIs for CPU 1
+ * ...
+ * + sysbus output IRQs: (in order; number will vary depending on number of
+ * cores)
+ * - IRQ for CPU 0
+ * - IRQ for CPU 1
+ * ...
+ * - FIQ for CPU 0
+ * - FIQ for CPU 1
+ * ...
+ * - VIRQ for CPU 0 (exists even if virt extensions not present)
+ * - VIRQ for CPU 1 (exists even if virt extensions not present)
+ * ...
+ * - VFIQ for CPU 0 (exists even if virt extensions not present)
+ * - VFIQ for CPU 1 (exists even if virt extensions not present)
+ * ...
+ * - maintenance IRQ for CPU i/f 0 (only if virt extensions present)
+ * - maintenance IRQ for CPU i/f 1 (only if virt extensions present)
*/
#define TYPE_CORTEX_MPCORE_PRIV "cortex_mpcore_priv"
OBJECT_DECLARE_TYPE(CortexMPPrivState, CortexMPPrivClass, CORTEX_MPCORE_PRIV)
@@ -41,8 +63,10 @@ OBJECT_DECLARE_TYPE(CortexMPPrivState, CortexMPPrivClass, CORTEX_MPCORE_PRIV)
/**
* CortexMPPrivClass:
* @container_size - size of the device's MMIO region
+ * @gic_class_name - GIC QOM class name
* @gic_spi_default - default number of GIC SPIs
* @gic_spi_max - maximum number of GIC SPIs
+ * @gic_revision - revision of the GIC
*/
struct CortexMPPrivClass {
SysBusDeviceClass parent_class;
@@ -51,14 +75,18 @@ struct CortexMPPrivClass {
uint64_t container_size;
+ const char *gic_class_name;
unsigned gic_spi_default;
unsigned gic_spi_max;
+ unsigned gic_revision;
+ uint32_t gic_priority_bits;
};
struct CortexMPPrivState {
SysBusDevice parent_obj;
MemoryRegion container;
+ GICState gic;
/* Properties */
uint32_t num_cores;
@@ -76,7 +104,6 @@ struct A9MPPrivState {
CortexMPPrivState parent_obj;
A9SCUState scu;
- GICState gic;
A9GTimerState gtimer;
ARMMPTimerState mptimer;
ARMMPTimerState wdt;
@@ -87,8 +114,6 @@ OBJECT_DECLARE_SIMPLE_TYPE(A15MPPrivState, A15MPCORE_PRIV)
struct A15MPPrivState {
CortexMPPrivState parent_obj;
-
- GICState gic;
};
#endif
@@ -22,34 +22,15 @@
#include "qapi/error.h"
#include "qemu/module.h"
#include "hw/cpu/cortex_mpcore.h"
-#include "hw/irq.h"
-#include "hw/qdev-properties.h"
-#include "sysemu/kvm.h"
-#include "kvm_arm.h"
-
-static void a15mp_priv_set_irq(void *opaque, int irq, int level)
-{
- A15MPPrivState *s = (A15MPPrivState *)opaque;
-
- qemu_set_irq(qdev_get_gpio_in(DEVICE(&s->gic), irq), level);
-}
-
-static void a15mp_priv_initfn(Object *obj)
-{
- A15MPPrivState *s = A15MPCORE_PRIV(obj);
-
- object_initialize_child(obj, "gic", &s->gic, gic_class_name());
- qdev_prop_set_uint32(DEVICE(&s->gic), "revision", 2);
-}
+#include "hw/core/cpu.h"
+#include "target/arm/cpu.h"
static void a15mp_priv_realize(DeviceState *dev, Error **errp)
{
CortexMPPrivClass *cc = CORTEX_MPCORE_PRIV_GET_CLASS(dev);
- SysBusDevice *sbd = SYS_BUS_DEVICE(dev);
- A15MPPrivState *s = A15MPCORE_PRIV(dev);
CortexMPPrivState *c = CORTEX_MPCORE_PRIV(dev);
- DeviceState *gicdev;
- SysBusDevice *gicsbd;
+ DeviceState *gicdev = DEVICE(&c->gic);
+ SysBusDevice *gicsbd = SYS_BUS_DEVICE(&c->gic);
Error *local_err = NULL;
int i;
@@ -59,32 +40,6 @@ static void a15mp_priv_realize(DeviceState *dev, Error **errp)
return;
}
- gicdev = DEVICE(&s->gic);
- qdev_prop_set_uint32(gicdev, "num-cpu", c->num_cores);
- qdev_prop_set_uint32(gicdev, "num-irq", c->gic_spi_num);
-
- if (!kvm_irqchip_in_kernel()) {
- /* Make the GIC's TZ support match the CPUs. We assume that
- * either all the CPUs have TZ, or none do.
- */
- qdev_prop_set_bit(gicdev, "has-security-extensions",
- c->cpu_has_el3);
- /* Similarly for virtualization support */
- qdev_prop_set_bit(gicdev, "has-virtualization-extensions",
- c->cpu_has_el2);
- }
-
- if (!sysbus_realize(SYS_BUS_DEVICE(&s->gic), errp)) {
- return;
- }
- gicsbd = SYS_BUS_DEVICE(&s->gic);
-
- /* Pass through outbound IRQ lines from the GIC */
- sysbus_pass_irq(sbd, gicsbd);
-
- /* Pass through inbound GPIO lines to the GIC */
- qdev_init_gpio_in(dev, a15mp_priv_set_irq, c->gic_spi_num - 32);
-
/* Wire the outputs from each CPU's generic timer to the
* appropriate GIC PPI inputs
*/
@@ -149,6 +104,8 @@ static void a15mp_priv_class_init(ObjectClass *klass, void *data)
cc->container_size = 0x8000;
+ cc->gic_class_name = gic_class_name();
+ cc->gic_revision = 2;
/*
* The Cortex-A15MP may have anything from 0 to 224 external interrupt
* IRQ lines (with another 32 internal). We default to 128+32, which
@@ -169,7 +126,6 @@ static const TypeInfo a15mp_types[] = {
.name = TYPE_A15MPCORE_PRIV,
.parent = TYPE_CORTEX_MPCORE_PRIV,
.instance_size = sizeof(A15MPPrivState),
- .instance_init = a15mp_priv_initfn,
.class_init = a15mp_priv_class_init,
},
};
@@ -12,28 +12,18 @@
#include "qapi/error.h"
#include "qemu/module.h"
#include "hw/cpu/cortex_mpcore.h"
-#include "hw/irq.h"
#include "hw/qdev-properties.h"
#include "hw/core/cpu.h"
#include "cpu.h"
#define A9_GIC_NUM_PRIORITY_BITS 5
-static void a9mp_priv_set_irq(void *opaque, int irq, int level)
-{
- A9MPPrivState *s = (A9MPPrivState *)opaque;
-
- qemu_set_irq(qdev_get_gpio_in(DEVICE(&s->gic), irq), level);
-}
-
static void a9mp_priv_initfn(Object *obj)
{
A9MPPrivState *s = A9MPCORE_PRIV(obj);
object_initialize_child(obj, "scu", &s->scu, TYPE_A9_SCU);
- object_initialize_child(obj, "gic", &s->gic, TYPE_ARM_GIC);
-
object_initialize_child(obj, "gtimer", &s->gtimer, TYPE_A9_GTIMER);
object_initialize_child(obj, "mptimer", &s->mptimer, TYPE_ARM_MPTIMER);
@@ -45,11 +35,11 @@ static void a9mp_priv_realize(DeviceState *dev, Error **errp)
{
CortexMPPrivClass *cc = CORTEX_MPCORE_PRIV_GET_CLASS(dev);
CortexMPPrivState *c = CORTEX_MPCORE_PRIV(dev);
- SysBusDevice *sbd = SYS_BUS_DEVICE(dev);
A9MPPrivState *s = A9MPCORE_PRIV(dev);
- DeviceState *scudev, *gicdev, *gtimerdev, *mptimerdev, *wdtdev;
- SysBusDevice *scubusdev, *gicbusdev, *gtimerbusdev, *mptimerbusdev,
- *wdtbusdev;
+ DeviceState *gicdev = DEVICE(&c->gic);
+ SysBusDevice *gicbusdev = SYS_BUS_DEVICE(&c->gic);
+ DeviceState *scudev, *gtimerdev, *mptimerdev, *wdtdev;
+ SysBusDevice *scubusdev, *gtimerbusdev, *mptimerbusdev, *wdtbusdev;
Error *local_err = NULL;
CPUState *cpu0;
Object *cpuobj;
@@ -76,28 +66,6 @@ static void a9mp_priv_realize(DeviceState *dev, Error **errp)
}
scubusdev = SYS_BUS_DEVICE(&s->scu);
- gicdev = DEVICE(&s->gic);
- qdev_prop_set_uint32(gicdev, "num-cpu", c->num_cores);
- qdev_prop_set_uint32(gicdev, "num-irq", c->gic_spi_num);
- qdev_prop_set_uint32(gicdev, "num-priority-bits",
- A9_GIC_NUM_PRIORITY_BITS);
-
- /* Make the GIC's TZ support match the CPUs. We assume that
- * either all the CPUs have TZ, or none do.
- */
- qdev_prop_set_bit(gicdev, "has-security-extensions", c->cpu_has_el3);
-
- if (!sysbus_realize(SYS_BUS_DEVICE(&s->gic), errp)) {
- return;
- }
- gicbusdev = SYS_BUS_DEVICE(&s->gic);
-
- /* Pass through outbound IRQ lines from the GIC */
- sysbus_pass_irq(sbd, gicbusdev);
-
- /* Pass through inbound GPIO lines to the GIC */
- qdev_init_gpio_in(dev, a9mp_priv_set_irq, c->gic_spi_num - 32);
-
gtimerdev = DEVICE(&s->gtimer);
qdev_prop_set_uint32(gtimerdev, "num-cpu", c->num_cores);
if (!sysbus_realize(SYS_BUS_DEVICE(&s->gtimer), errp)) {
@@ -167,6 +135,9 @@ static void a9mp_priv_class_init(ObjectClass *klass, void *data)
cc->container_size = 0x2000;
+ cc->gic_class_name = TYPE_ARM_GIC;
+ cc->gic_revision = 1;
+ cc->gic_priority_bits = A9_GIC_NUM_PRIORITY_BITS;
/*
* The Cortex-A9MP may have anything from 0 to 224 external interrupt
* IRQ lines (with another 32 internal). We default to 64+32, which
@@ -10,6 +10,15 @@
#include "qapi/error.h"
#include "hw/qdev-properties.h"
#include "hw/cpu/cortex_mpcore.h"
+#include "hw/irq.h"
+#include "sysemu/kvm.h"
+
+static void cortex_mpcore_priv_set_irq(void *opaque, int irq, int level)
+{
+ CortexMPPrivState *s = (CortexMPPrivState *)opaque;
+
+ qemu_set_irq(qdev_get_gpio_in(DEVICE(&s->gic), irq), level);
+}
static void cortex_mpcore_priv_instance_init(Object *obj)
{
@@ -23,6 +32,42 @@ static void cortex_mpcore_priv_instance_init(Object *obj)
sysbus_init_mmio(sbd, &s->container);
s->gic_spi_num = k->gic_spi_default;
+ object_initialize_child(obj, "gic", &s->gic, k->gic_class_name);
+ qdev_prop_set_uint32(DEVICE(&s->gic), "revision", k->gic_revision);
+}
+
+static void cortex_mpcore_priv_realize(DeviceState *dev, Error **errp)
+{
+ SysBusDevice *sbd = SYS_BUS_DEVICE(dev);
+ CortexMPPrivState *s = CORTEX_MPCORE_PRIV(dev);
+ CortexMPPrivClass *k = CORTEX_MPCORE_PRIV_GET_CLASS(dev);
+ DeviceState *gicdev = DEVICE(&s->gic);
+
+ qdev_prop_set_uint32(gicdev, "num-cpu", s->num_cores);
+ qdev_prop_set_uint32(gicdev, "num-irq", s->gic_spi_num);
+ if (k->gic_priority_bits) {
+ qdev_prop_set_uint32(gicdev, "num-priority-bits", k->gic_priority_bits);
+ }
+ if (!kvm_irqchip_in_kernel()) {
+ /*
+ * Make the GIC's TZ support match the CPUs. We assume that
+ * either all the CPUs have TZ, or none do.
+ */
+ qdev_prop_set_bit(gicdev, "has-security-extensions",
+ s->cpu_has_el3);
+ /* Similarly for virtualization support */
+ qdev_prop_set_bit(gicdev, "has-virtualization-extensions",
+ s->cpu_has_el2);
+ }
+ if (!sysbus_realize(SYS_BUS_DEVICE(&s->gic), errp)) {
+ return;
+ }
+
+ /* Pass through outbound IRQ lines from the GIC */
+ sysbus_pass_irq(sbd, SYS_BUS_DEVICE(&s->gic));
+
+ /* Pass through inbound GPIO lines to the GIC */
+ qdev_init_gpio_in(dev, cortex_mpcore_priv_set_irq, s->gic_spi_num - 32);
}
static Property cortex_mpcore_priv_properties[] = {
@@ -42,6 +87,7 @@ static void cortex_mpcore_priv_class_init(ObjectClass *klass, void *data)
{
DeviceClass *dc = DEVICE_CLASS(klass);
+ dc->realize = cortex_mpcore_priv_realize;
device_class_set_props(dc, cortex_mpcore_priv_properties);
/* We currently have no saveable state */
}
@@ -645,19 +645,18 @@ config ZAURUS
config CORTEX_MPCORE
bool
+ select ARM_GIC
config A9MPCORE
bool
select CORTEX_MPCORE
select A9_GTIMER
select A9SCU # snoop control unit
- select ARM_GIC
select ARM_MPTIMER
config A15MPCORE
bool
select CORTEX_MPCORE
- select ARM_GIC
config ARM11MPCORE
bool
Move the GIC instance creation to the abstract QOM parent. Signed-off-by: Philippe Mathieu-Daudé <philmd@linaro.org> --- include/hw/cpu/cortex_mpcore.h | 31 +++++++++++++++++-- hw/cpu/a15mpcore.c | 56 ++++------------------------------ hw/cpu/a9mpcore.c | 43 +++++--------------------- hw/cpu/cortex_mpcore.c | 46 ++++++++++++++++++++++++++++ hw/arm/Kconfig | 3 +- 5 files changed, 88 insertions(+), 91 deletions(-)