|
@@ -352,6 +352,53 @@ static uint32_t maintenance_interrupt_state(GICv3CPUState *cs)
|
|
|
|
|
|
static void gicv3_cpuif_virt_update(GICv3CPUState *cs)
|
|
|
{
|
|
|
+ /* Tell the CPU about any pending virtual interrupts or
|
|
|
+ * maintenance interrupts, following a change to the state
|
|
|
+ * of the CPU interface relevant to virtual interrupts.
|
|
|
+ *
|
|
|
+ * CAUTION: this function will call qemu_set_irq() on the
|
|
|
+ * CPU maintenance IRQ line, which is typically wired up
|
|
|
+ * to the GIC as a per-CPU interrupt. This means that it
|
|
|
+ * will recursively call back into the GIC code via
|
|
|
+ * gicv3_redist_set_irq() and thus into the CPU interface code's
|
|
|
+ * gicv3_cpuif_update(). It is therefore important that this
|
|
|
+ * function is only called as the final action of a CPU interface
|
|
|
+ * register write implementation, after all the GIC state
|
|
|
+ * fields have been updated. gicv3_cpuif_update() also must
|
|
|
+ * not cause this function to be called, but that happens
|
|
|
+ * naturally as a result of there being no architectural
|
|
|
+ * linkage between the physical and virtual GIC logic.
|
|
|
+ */
|
|
|
+ int idx;
|
|
|
+ int irqlevel = 0;
|
|
|
+ int fiqlevel = 0;
|
|
|
+ int maintlevel = 0;
|
|
|
+
|
|
|
+ idx = hppvi_index(cs);
|
|
|
+ trace_gicv3_cpuif_virt_update(gicv3_redist_affid(cs), idx);
|
|
|
+ if (idx >= 0) {
|
|
|
+ uint64_t lr = cs->ich_lr_el2[idx];
|
|
|
+
|
|
|
+ if (icv_hppi_can_preempt(cs, lr)) {
|
|
|
+ /* Virtual interrupts are simple: G0 are always FIQ, and G1 IRQ */
|
|
|
+ if (lr & ICH_LR_EL2_GROUP) {
|
|
|
+ irqlevel = 1;
|
|
|
+ } else {
|
|
|
+ fiqlevel = 1;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ if (cs->ich_hcr_el2 & ICH_HCR_EL2_EN) {
|
|
|
+ maintlevel = maintenance_interrupt_state(cs);
|
|
|
+ }
|
|
|
+
|
|
|
+ trace_gicv3_cpuif_virt_set_irqs(gicv3_redist_affid(cs), fiqlevel,
|
|
|
+ irqlevel, maintlevel);
|
|
|
+
|
|
|
+ qemu_set_irq(cs->parent_vfiq, fiqlevel);
|
|
|
+ qemu_set_irq(cs->parent_virq, irqlevel);
|
|
|
+ qemu_set_irq(cs->maintenance_irq, maintlevel);
|
|
|
}
|
|
|
|
|
|
static uint64_t icv_ap_read(CPUARMState *env, const ARMCPRegInfo *ri)
|
|
@@ -2480,6 +2527,8 @@ void gicv3_init_cpuif(GICv3State *s)
|
|
|
&& cpu->gic_num_lrs) {
|
|
|
int j;
|
|
|
|
|
|
+ cs->maintenance_irq = cpu->gicv3_maintenance_interrupt;
|
|
|
+
|
|
|
cs->num_list_regs = cpu->gic_num_lrs;
|
|
|
cs->vpribits = cpu->gic_vpribits;
|
|
|
cs->vprebits = cpu->gic_vprebits;
|