On 04/24/2018 08:51 AM, David Gibson wrote: > On Thu, Apr 19, 2018 at 02:43:00PM +0200, Cédric Le Goater wrote: >> sPAPRXive is a model for the XIVE interrupt controller device of the >> sPAPR machine. It holds the routing XIVE table, the Interrupt >> Virtualization Entry (IVE) table which associates interrupt source >> numbers with targets. >> >> Also extend the XiveFabric with an accessor to the IVT. This will be >> needed by the routing algorithm. >> >> Signed-off-by: Cédric Le Goater <c...@kaod.org> >> --- >> >> May be should introduce a XiveRouter model to hold the IVT. To be >> discussed. > > Yeah, maybe. Am I correct in thinking that on pnv there could be more > than one XiveRouter?
There is only one, the main IC. > If we did have a XiveRouter, I'm not sure we'd need the XiveFabric > interface, possibly its methods could just be class methods of > XiveRouter. Yes. We could introduce a XiveRouter to share the ivt table between the sPAPRXive and the PnvXIVE models, the interrupt controllers of the machines. Methods would provide way to get the ivt/eq/nvt objects required for routing. I need to add a set_eq() to push the EQ data. The XiveRouter would also be a XiveFabric (or some other name) to let the internal sources of the interrupt controller forward events. >> >> Changes since v2 : >> >> - introduced the XiveFabric interface >> >> default-configs/ppc64-softmmu.mak | 1 + >> hw/intc/Makefile.objs | 1 + >> hw/intc/spapr_xive.c | 159 >> ++++++++++++++++++++++++++++++++++++++ >> hw/intc/xive.c | 7 ++ >> include/hw/ppc/spapr_xive.h | 31 ++++++++ >> include/hw/ppc/xive.h | 5 ++ >> include/hw/ppc/xive_regs.h | 33 ++++++++ >> 7 files changed, 237 insertions(+) >> create mode 100644 hw/intc/spapr_xive.c >> create mode 100644 include/hw/ppc/spapr_xive.h >> create mode 100644 include/hw/ppc/xive_regs.h >> >> diff --git a/default-configs/ppc64-softmmu.mak >> b/default-configs/ppc64-softmmu.mak >> index c6d13e757977..f8d34722931d 100644 >> --- a/default-configs/ppc64-softmmu.mak >> +++ b/default-configs/ppc64-softmmu.mak >> @@ -17,4 +17,5 @@ CONFIG_XICS=$(CONFIG_PSERIES) >> CONFIG_XICS_SPAPR=$(CONFIG_PSERIES) >> CONFIG_XICS_KVM=$(call land,$(CONFIG_PSERIES),$(CONFIG_KVM)) >> CONFIG_XIVE=$(CONFIG_PSERIES) >> +CONFIG_XIVE_SPAPR=$(CONFIG_PSERIES) >> CONFIG_MEM_HOTPLUG=y >> diff --git a/hw/intc/Makefile.objs b/hw/intc/Makefile.objs >> index 72a46ed91c31..301a8e972d91 100644 >> --- a/hw/intc/Makefile.objs >> +++ b/hw/intc/Makefile.objs >> @@ -38,6 +38,7 @@ obj-$(CONFIG_XICS) += xics.o >> obj-$(CONFIG_XICS_SPAPR) += xics_spapr.o >> obj-$(CONFIG_XICS_KVM) += xics_kvm.o >> obj-$(CONFIG_XIVE) += xive.o >> +obj-$(CONFIG_XIVE_SPAPR) += spapr_xive.o >> obj-$(CONFIG_POWERNV) += xics_pnv.o >> obj-$(CONFIG_ALLWINNER_A10_PIC) += allwinner-a10-pic.o >> obj-$(CONFIG_S390_FLIC) += s390_flic.o >> diff --git a/hw/intc/spapr_xive.c b/hw/intc/spapr_xive.c >> new file mode 100644 >> index 000000000000..020444e2665a >> --- /dev/null >> +++ b/hw/intc/spapr_xive.c >> @@ -0,0 +1,159 @@ >> +/* >> + * QEMU PowerPC sPAPR XIVE interrupt controller model >> + * >> + * Copyright (c) 2017-2018, IBM Corporation. >> + * >> + * This code is licensed under the GPL version 2 or later. See the >> + * COPYING file in the top-level directory. >> + */ >> + >> +#include "qemu/osdep.h" >> +#include "qemu/log.h" >> +#include "qapi/error.h" >> +#include "target/ppc/cpu.h" >> +#include "sysemu/cpus.h" >> +#include "monitor/monitor.h" >> +#include "hw/ppc/spapr_xive.h" >> +#include "hw/ppc/xive_regs.h" >> + >> +void spapr_xive_pic_print_info(sPAPRXive *xive, Monitor *mon) >> +{ >> + int i; >> + >> + monitor_printf(mon, "IVE Table\n"); >> + for (i = 0; i < xive->nr_irqs; i++) { >> + XiveIVE *ive = &xive->ivt[i]; >> + >> + if (!(ive->w & IVE_VALID)) { >> + continue; >> + } >> + >> + monitor_printf(mon, " %4x %s %08x %08x\n", i, >> + ive->w & IVE_MASKED ? "M" : " ", >> + (int) GETFIELD(IVE_EQ_INDEX, ive->w), >> + (int) GETFIELD(IVE_EQ_DATA, ive->w)); >> + } >> +} >> + >> +static void spapr_xive_reset(DeviceState *dev) >> +{ >> + sPAPRXive *xive = SPAPR_XIVE(dev); >> + int i; >> + >> + /* Mask all valid IVEs in the IRQ number space. */ >> + for (i = 0; i < xive->nr_irqs; i++) { >> + XiveIVE *ive = &xive->ivt[i]; >> + if (ive->w & IVE_VALID) { >> + ive->w |= IVE_MASKED; >> + } >> + } >> +} >> + >> +static void spapr_xive_init(Object *obj) > > I'm trying to standardize on init_instance methods being called > *_instance_init(). It helps to make it obvious that this is ineed an > instance_init() method, rather than one of the various other init > calls that exist in various places. ok. this is good practice. I will fix. Thanks, C. > >> +{ >> + >> +} >> + >> +static void spapr_xive_realize(DeviceState *dev, Error **errp) >> +{ >> + sPAPRXive *xive = SPAPR_XIVE(dev); >> + >> + if (!xive->nr_irqs) { >> + error_setg(errp, "Number of interrupt needs to be greater 0"); >> + return; >> + } >> + >> + /* Allocate the Interrupt Virtualization Table */ >> + xive->ivt = g_new0(XiveIVE, xive->nr_irqs); >> +} >> + >> +static XiveIVE *spapr_xive_get_ive(XiveFabric *xf, uint32_t lisn) >> +{ >> + sPAPRXive *xive = SPAPR_XIVE(xf); >> + >> + return lisn < xive->nr_irqs ? &xive->ivt[lisn] : NULL; >> +} >> + >> +static const VMStateDescription vmstate_spapr_xive_ive = { >> + .name = TYPE_SPAPR_XIVE "/ive", >> + .version_id = 1, >> + .minimum_version_id = 1, >> + .fields = (VMStateField []) { >> + VMSTATE_UINT64(w, XiveIVE), >> + VMSTATE_END_OF_LIST() >> + }, >> +}; >> + >> +static const VMStateDescription vmstate_spapr_xive = { >> + .name = TYPE_SPAPR_XIVE, >> + .version_id = 1, >> + .minimum_version_id = 1, >> + .fields = (VMStateField[]) { >> + VMSTATE_UINT32_EQUAL(nr_irqs, sPAPRXive, NULL), >> + VMSTATE_STRUCT_VARRAY_POINTER_UINT32(ivt, sPAPRXive, nr_irqs, >> + vmstate_spapr_xive_ive, XiveIVE), >> + VMSTATE_END_OF_LIST() >> + }, >> +}; >> + >> +static Property spapr_xive_properties[] = { >> + DEFINE_PROP_UINT32("nr-irqs", sPAPRXive, nr_irqs, 0), >> + DEFINE_PROP_END_OF_LIST(), >> +}; >> + >> +static void spapr_xive_class_init(ObjectClass *klass, void *data) >> +{ >> + DeviceClass *dc = DEVICE_CLASS(klass); >> + XiveFabricClass *xfc = XIVE_FABRIC_CLASS(klass); >> + >> + dc->realize = spapr_xive_realize; >> + dc->reset = spapr_xive_reset; >> + dc->props = spapr_xive_properties; >> + dc->desc = "sPAPR XIVE interrupt controller"; >> + dc->vmsd = &vmstate_spapr_xive; >> + >> + xfc->get_ive = spapr_xive_get_ive; >> +} >> + >> +static const TypeInfo spapr_xive_info = { >> + .name = TYPE_SPAPR_XIVE, >> + .parent = TYPE_SYS_BUS_DEVICE, >> + .instance_init = spapr_xive_init, >> + .instance_size = sizeof(sPAPRXive), >> + .class_init = spapr_xive_class_init, >> + .interfaces = (InterfaceInfo[]) { >> + { TYPE_XIVE_FABRIC }, >> + { }, >> + }, >> +}; >> + >> +static void spapr_xive_register_types(void) >> +{ >> + type_register_static(&spapr_xive_info); >> +} >> + >> +type_init(spapr_xive_register_types) >> + >> +bool spapr_xive_irq_enable(sPAPRXive *xive, uint32_t lisn, bool lsi) >> +{ >> + XiveIVE *ive = spapr_xive_get_ive(XIVE_FABRIC(xive), lisn); >> + >> + if (!ive) { >> + return false; >> + } >> + >> + ive->w |= IVE_VALID; >> + return true; >> +} >> + >> +bool spapr_xive_irq_disable(sPAPRXive *xive, uint32_t lisn) >> +{ >> + XiveIVE *ive = spapr_xive_get_ive(XIVE_FABRIC(xive), lisn); >> + >> + if (!ive) { >> + return false; >> + } >> + >> + ive->w &= ~IVE_VALID; >> + return true; >> +} >> diff --git a/hw/intc/xive.c b/hw/intc/xive.c >> index b4c3d06c1219..dccad0318834 100644 >> --- a/hw/intc/xive.c >> +++ b/hw/intc/xive.c >> @@ -20,6 +20,13 @@ >> * XIVE Fabric >> */ >> >> +XiveIVE *xive_fabric_get_ive(XiveFabric *xf, uint32_t lisn) >> +{ >> + XiveFabricClass *xfc = XIVE_FABRIC_GET_CLASS(xf); >> + >> + return xfc->get_ive(xf, lisn); >> +} >> + >> static void xive_fabric_route(XiveFabric *xf, int lisn) >> { >> >> diff --git a/include/hw/ppc/spapr_xive.h b/include/hw/ppc/spapr_xive.h >> new file mode 100644 >> index 000000000000..1d966b5d3a96 >> --- /dev/null >> +++ b/include/hw/ppc/spapr_xive.h >> @@ -0,0 +1,31 @@ >> +/* >> + * QEMU PowerPC sPAPR XIVE interrupt controller model >> + * >> + * Copyright (c) 2017-2018, IBM Corporation. >> + * >> + * This code is licensed under the GPL version 2 or later. See the >> + * COPYING file in the top-level directory. >> + */ >> + >> +#ifndef PPC_SPAPR_XIVE_H >> +#define PPC_SPAPR_XIVE_H >> + >> +#include "hw/sysbus.h" >> +#include "hw/ppc/xive.h" >> + >> +#define TYPE_SPAPR_XIVE "spapr-xive" >> +#define SPAPR_XIVE(obj) OBJECT_CHECK(sPAPRXive, (obj), TYPE_SPAPR_XIVE) >> + >> +typedef struct sPAPRXive { >> + SysBusDevice parent; >> + >> + /* Routing table */ >> + XiveIVE *ivt; >> + uint32_t nr_irqs; >> +} sPAPRXive; >> + >> +bool spapr_xive_irq_enable(sPAPRXive *xive, uint32_t lisn, bool lsi); >> +bool spapr_xive_irq_disable(sPAPRXive *xive, uint32_t lisn); >> +void spapr_xive_pic_print_info(sPAPRXive *xive, Monitor *mon); >> + >> +#endif /* PPC_SPAPR_XIVE_H */ >> diff --git a/include/hw/ppc/xive.h b/include/hw/ppc/xive.h >> index 4fcae2c763e6..5b145816acdc 100644 >> --- a/include/hw/ppc/xive.h >> +++ b/include/hw/ppc/xive.h >> @@ -11,6 +11,7 @@ >> #define PPC_XIVE_H >> >> #include "hw/sysbus.h" >> +#include "hw/ppc/xive_regs.h" >> >> typedef struct XiveFabric XiveFabric; >> >> @@ -166,6 +167,10 @@ typedef struct XiveFabric { >> typedef struct XiveFabricClass { >> InterfaceClass parent; >> void (*notify)(XiveFabric *xf, uint32_t lisn); >> + >> + XiveIVE *(*get_ive)(XiveFabric *xf, uint32_t lisn); >> } XiveFabricClass; >> >> +XiveIVE *xive_fabric_get_ive(XiveFabric *xf, uint32_t lisn); >> + >> #endif /* PPC_XIVE_H */ >> diff --git a/include/hw/ppc/xive_regs.h b/include/hw/ppc/xive_regs.h >> new file mode 100644 >> index 000000000000..5903f29eb789 >> --- /dev/null >> +++ b/include/hw/ppc/xive_regs.h >> @@ -0,0 +1,33 @@ >> +/* >> + * QEMU PowerPC XIVE interrupt controller model >> + * >> + * Copyright (c) 2016-2018, IBM Corporation. >> + * >> + * This code is licensed under the GPL version 2 or later. See the >> + * COPYING file in the top-level directory. >> + */ >> + >> +#ifndef _PPC_XIVE_REGS_H >> +#define _PPC_XIVE_REGS_H >> + >> +/* IVE/EAS >> + * >> + * One per interrupt source. Targets that interrupt to a given EQ >> + * and provides the corresponding logical interrupt number (EQ data) >> + * >> + * We also map this structure to the escalation descriptor inside >> + * an EQ, though in that case the valid and masked bits are not used. >> + */ >> +typedef struct XiveIVE { >> + /* Use a single 64-bit definition to make it easier to >> + * perform atomic updates >> + */ >> + uint64_t w; >> +#define IVE_VALID PPC_BIT(0) >> +#define IVE_EQ_BLOCK PPC_BITMASK(4, 7) /* Destination EQ block# */ >> +#define IVE_EQ_INDEX PPC_BITMASK(8, 31) /* Destination EQ index */ >> +#define IVE_MASKED PPC_BIT(32) /* Masked */ >> +#define IVE_EQ_DATA PPC_BITMASK(33, 63) /* Data written to the EQ >> */ >> +} XiveIVE; >> + >> +#endif /* _INTC_XIVE_INTERNAL_H */ >