提交 2e2a1b46 编写于 作者: P Paul Burton 提交者: Yongbok Kim

hw/mips_gic: Update pin state on mask changes

If the GIC interrupt mask is changed by a write to the smask (set mask)
or rmask (reset mask) registers, we need to re-evaluate the state of the
pins/IRQs fed to the CPU. Without doing so we risk leaving a pin high
despite the interrupt that led to that state being masked, or losing
interrupts if an already pending interrupt is unmasked.
Signed-off-by: NPaul Burton <paul.burton@imgtec.com>
Reviewed-by: NLeon Alrae <leon.alrae@imgtec.com>
Signed-off-by: NYongbok Kim <yongbok.kim@imgtec.com>
上级 eb90ab94
...@@ -20,31 +20,29 @@ ...@@ -20,31 +20,29 @@
#include "kvm_mips.h" #include "kvm_mips.h"
#include "hw/intc/mips_gic.h" #include "hw/intc/mips_gic.h"
static void mips_gic_set_vp_irq(MIPSGICState *gic, int vp, int pin, int level) static void mips_gic_set_vp_irq(MIPSGICState *gic, int vp, int pin)
{ {
int ored_level = level; int ored_level = 0;
int i; int i;
/* ORing pending registers sharing same pin */ /* ORing pending registers sharing same pin */
if (!ored_level) { for (i = 0; i < gic->num_irq; i++) {
for (i = 0; i < gic->num_irq; i++) { if ((gic->irq_state[i].map_pin & GIC_MAP_MSK) == pin &&
if ((gic->irq_state[i].map_pin & GIC_MAP_MSK) == pin && gic->irq_state[i].map_vp == vp &&
gic->irq_state[i].map_vp == vp && gic->irq_state[i].enabled) {
gic->irq_state[i].enabled) { ored_level |= gic->irq_state[i].pending;
ored_level |= gic->irq_state[i].pending;
}
if (ored_level) {
/* no need to iterate all interrupts */
break;
}
} }
if (((gic->vps[vp].compare_map & GIC_MAP_MSK) == pin) && if (ored_level) {
(gic->vps[vp].mask & GIC_VP_MASK_CMP_MSK)) { /* no need to iterate all interrupts */
/* ORing with local pending register (count/compare) */ break;
ored_level |= (gic->vps[vp].pend & GIC_VP_MASK_CMP_MSK) >>
GIC_VP_MASK_CMP_SHF;
} }
} }
if (((gic->vps[vp].compare_map & GIC_MAP_MSK) == pin) &&
(gic->vps[vp].mask & GIC_VP_MASK_CMP_MSK)) {
/* ORing with local pending register (count/compare) */
ored_level |= (gic->vps[vp].pend & GIC_VP_MASK_CMP_MSK) >>
GIC_VP_MASK_CMP_SHF;
}
if (kvm_enabled()) { if (kvm_enabled()) {
kvm_mips_set_ipi_interrupt(mips_env_get_cpu(gic->vps[vp].env), kvm_mips_set_ipi_interrupt(mips_env_get_cpu(gic->vps[vp].env),
pin + GIC_CPU_PIN_OFFSET, pin + GIC_CPU_PIN_OFFSET,
...@@ -55,21 +53,27 @@ static void mips_gic_set_vp_irq(MIPSGICState *gic, int vp, int pin, int level) ...@@ -55,21 +53,27 @@ static void mips_gic_set_vp_irq(MIPSGICState *gic, int vp, int pin, int level)
} }
} }
static void gic_set_irq(void *opaque, int n_IRQ, int level) static void gic_update_pin_for_irq(MIPSGICState *gic, int n_IRQ)
{ {
MIPSGICState *gic = (MIPSGICState *) opaque;
int vp = gic->irq_state[n_IRQ].map_vp; int vp = gic->irq_state[n_IRQ].map_vp;
int pin = gic->irq_state[n_IRQ].map_pin & GIC_MAP_MSK; int pin = gic->irq_state[n_IRQ].map_pin & GIC_MAP_MSK;
if (vp < 0 || vp >= gic->num_vps) {
return;
}
mips_gic_set_vp_irq(gic, vp, pin);
}
static void gic_set_irq(void *opaque, int n_IRQ, int level)
{
MIPSGICState *gic = (MIPSGICState *) opaque;
gic->irq_state[n_IRQ].pending = (uint8_t) level; gic->irq_state[n_IRQ].pending = (uint8_t) level;
if (!gic->irq_state[n_IRQ].enabled) { if (!gic->irq_state[n_IRQ].enabled) {
/* GIC interrupt source disabled */ /* GIC interrupt source disabled */
return; return;
} }
if (vp < 0 || vp >= gic->num_vps) { gic_update_pin_for_irq(gic, n_IRQ);
return;
}
mips_gic_set_vp_irq(gic, vp, pin, level);
} }
#define OFFSET_CHECK(c) \ #define OFFSET_CHECK(c) \
...@@ -209,7 +213,7 @@ static void gic_timer_store_vp_compare(MIPSGICState *gic, uint32_t vp_index, ...@@ -209,7 +213,7 @@ static void gic_timer_store_vp_compare(MIPSGICState *gic, uint32_t vp_index,
gic->vps[vp_index].pend &= ~(1 << GIC_LOCAL_INT_COMPARE); gic->vps[vp_index].pend &= ~(1 << GIC_LOCAL_INT_COMPARE);
if (gic->vps[vp_index].compare_map & GIC_MAP_TO_PIN_MSK) { if (gic->vps[vp_index].compare_map & GIC_MAP_TO_PIN_MSK) {
uint32_t pin = (gic->vps[vp_index].compare_map & GIC_MAP_MSK); uint32_t pin = (gic->vps[vp_index].compare_map & GIC_MAP_MSK);
mips_gic_set_vp_irq(gic, vp_index, pin, 0); mips_gic_set_vp_irq(gic, vp_index, pin);
} }
mips_gictimer_store_vp_compare(gic->gic_timer, vp_index, compare); mips_gictimer_store_vp_compare(gic->gic_timer, vp_index, compare);
} }
...@@ -286,6 +290,7 @@ static void gic_write(void *opaque, hwaddr addr, uint64_t data, unsigned size) ...@@ -286,6 +290,7 @@ static void gic_write(void *opaque, hwaddr addr, uint64_t data, unsigned size)
OFFSET_CHECK((base + size * 8) <= gic->num_irq); OFFSET_CHECK((base + size * 8) <= gic->num_irq);
for (i = 0; i < size * 8; i++) { for (i = 0; i < size * 8; i++) {
gic->irq_state[base + i].enabled &= !((data >> i) & 1); gic->irq_state[base + i].enabled &= !((data >> i) & 1);
gic_update_pin_for_irq(gic, base + i);
} }
break; break;
case GIC_SH_WEDGE_OFS: case GIC_SH_WEDGE_OFS:
...@@ -305,6 +310,7 @@ static void gic_write(void *opaque, hwaddr addr, uint64_t data, unsigned size) ...@@ -305,6 +310,7 @@ static void gic_write(void *opaque, hwaddr addr, uint64_t data, unsigned size)
OFFSET_CHECK((base + size * 8) <= gic->num_irq); OFFSET_CHECK((base + size * 8) <= gic->num_irq);
for (i = 0; i < size * 8; i++) { for (i = 0; i < size * 8; i++) {
gic->irq_state[base + i].enabled |= (data >> i) & 1; gic->irq_state[base + i].enabled |= (data >> i) & 1;
gic_update_pin_for_irq(gic, base + i);
} }
break; break;
case GIC_SH_MAP0_PIN_OFS ... GIC_SH_MAP255_PIN_OFS: case GIC_SH_MAP0_PIN_OFS ... GIC_SH_MAP255_PIN_OFS:
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册