[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index] [Xen-devel] [RFC 22/22] xen/arm: gic-v3: Add support of vGICv2 when available
* Modify the GICv3 driver to recognize a such device. I wasn't able to find a register which tell if GICv2 is supported on GICv3. The only way to find it seems to check if the DT node provides GICC and GICV. * Disable access to ICC_SRE_EL1 to guest using vGICv2 * The LR is slightly different for vGICv2. The interrupt is always injected with group0. * Add a comment explaining why Group1 is used for vGICv3. Signed-off-by: Julien Grall <julien.grall@xxxxxxxxxx> --- xen/arch/arm/gic-v3.c | 58 ++++++++++++++++++++++++++++++++++++++++++++++----- 1 file changed, 53 insertions(+), 5 deletions(-) diff --git a/xen/arch/arm/gic-v3.c b/xen/arch/arm/gic-v3.c index 329d6ca..8533ae5 100644 --- a/xen/arch/arm/gic-v3.c +++ b/xen/arch/arm/gic-v3.c @@ -237,15 +237,14 @@ static void gicv3_ich_write_lr(int lr, uint64_t val) } /* - * System Register Enable (SRE). Enable to access CPU & Virtual - * interface registers as system registers in EL2 + * System Register Enable (SRE). */ static void gicv3_enable_sre(void) { uint32_t val; val = READ_SYSREG32(ICC_SRE_EL2); - val |= GICC_SRE_EL2_SRE | GICC_SRE_EL2_ENEL1; + val |= GICC_SRE_EL2_SRE; WRITE_SYSREG32(val, ICC_SRE_EL2); isb(); @@ -373,6 +372,20 @@ static void gicv3_save_state(struct vcpu *v) static void gicv3_restore_state(const struct vcpu *v) { + uint32_t val; + + val = READ_SYSREG32(ICC_SRE_EL2); + /* + * Don't give access to system registers when the guest is using + * GICv2 + */ + if ( v->domain->arch.vgic.version == GIC_V2 ) + val &= ~GICC_SRE_EL2_ENEL1; + else + val |= GICC_SRE_EL2_ENEL1; + WRITE_SYSREG32(val, ICC_SRE_EL2); + isb(); + WRITE_SYSREG32(v->arch.gic.v3.sre_el1, ICC_SRE_EL1); WRITE_SYSREG32(v->arch.gic.v3.vmcr, ICH_VMCR_EL2); restore_aprn_regs(&v->arch.gic); @@ -843,13 +856,20 @@ static void gicv3_disable_interface(void) static void gicv3_update_lr(int lr, const struct pending_irq *p, unsigned int state) { - uint64_t grp = GICH_LR_GRP1; uint64_t val = 0; BUG_ON(lr >= gicv3_info.nr_lrs); BUG_ON(lr < 0); - val = (((uint64_t)state & 0x3) << GICH_LR_STATE_SHIFT) | grp; + val = (((uint64_t)state & 0x3) << GICH_LR_STATE_SHIFT); + + /* + * When the guest is GICv3, all guest IRQs are Group 1, as Group0 + * would result in a FIQ in the guest, which it wouldn't expect + */ + if ( current->domain->arch.vgic.version == GIC_V3 ) + val |= GICH_LR_GRP1; + val |= ((uint64_t)p->priority & 0xff) << GICH_LR_PRIORITY_SHIFT; val |= ((uint64_t)p->irq & GICH_LR_VIRTUAL_MASK) << GICH_LR_VIRTUAL_SHIFT; @@ -1107,6 +1127,32 @@ static int __init cmp_rdist(const void *a, const void *b) return ( l->base < r->base) ? -1 : 0; } +/* If the GICv3 supports GICv2, initialize it */ +static void __init gicv3_init_v2(const struct dt_device_node *node) +{ + int res; + + /* + * For GICv3 supporting GICv2, GICC and GICV base address will be + * provided. + */ + + res = dt_device_get_address(node, 1 + gicv3_info.nr_rdist_regions, + &gicv3_info.cbase, NULL); + if ( res ) + return; + + res = dt_device_get_address(node, 1 + gicv3_info.nr_rdist_regions + 2, + &gicv3_info.vbase, NULL); + if ( res ) + return; + + printk("GICv3 compatible with GICv2 cbase %#"PRIpaddr" vbase %#"PRIpaddr"\n", + gicv3_info.cbase, gicv3_info.vbase); + + gicv3_info.vgic_versions |= GIC_V2; +} + /* Set up the GIC */ static int __init gicv3_init(void) { @@ -1208,6 +1254,8 @@ static int __init gicv3_init(void) i, r->base, r->base + r->size); } + gicv3_init_v2(node); + spin_lock_init(&gicv3.lock); spin_lock(&gicv3.lock); -- 2.1.4 _______________________________________________ Xen-devel mailing list Xen-devel@xxxxxxxxxxxxx http://lists.xen.org/xen-devel
|
Lists.xenproject.org is hosted with RackSpace, monitoring our |