aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/irqchip/irq-gic-v3.c
diff options
context:
space:
mode:
authorMarc Zyngier <marc.zyngier@arm.com>2015-08-26 12:00:43 -0400
committerThomas Gleixner <tglx@linutronix.de>2015-08-27 11:13:49 -0400
commit530bf353e4eb06bcba5078390c949650cd26a7c7 (patch)
treed18dd4acfce81aa7b67c9c65cd8c0d5c8645e0b6 /drivers/irqchip/irq-gic-v3.c
parent0b6a3da9617a08e13afc09cb7e148470ed0eb280 (diff)
irqchip/GICv3: Don't deactivate interrupts forwarded to a guest
Commit 0a4377de3056 ("genirq: Introduce irq_set_vcpu_affinity() to target an interrupt to a VCPU") added just what we needed at the lowest level to allow an interrupt to be deactivated by a guest. When such a request reaches the GIC, it knows it doesn't need to perform the deactivation anymore, and can safely leave the guest do its magic. This of course requires additional support in both VFIO and KVM. Signed-off-by: Marc Zyngier <marc.zyngier@arm.com> Reviewed-and-tested-by: Eric Auger <eric.auger@linaro.org> Cc: Christoffer Dall <christoffer.dall@linaro.org> Cc: Jiang Liu <jiang.liu@linux.intel.com> Cc: <linux-arm-kernel@lists.infradead.org> Cc: kvmarm@lists.cs.columbia.edu Cc: Jason Cooper <jason@lakedaemon.net> Link: http://lkml.kernel.org/r/1440604845-28229-3-git-send-email-marc.zyngier@arm.com Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Diffstat (limited to 'drivers/irqchip/irq-gic-v3.c')
-rw-r--r--drivers/irqchip/irq-gic-v3.c27
1 files changed, 25 insertions, 2 deletions
diff --git a/drivers/irqchip/irq-gic-v3.c b/drivers/irqchip/irq-gic-v3.c
index 5c31cc9353d5..7deed6ef54c2 100644
--- a/drivers/irqchip/irq-gic-v3.c
+++ b/drivers/irqchip/irq-gic-v3.c
@@ -70,6 +70,11 @@ static inline int gic_irq_in_rdist(struct irq_data *d)
70 return gic_irq(d) < 32; 70 return gic_irq(d) < 32;
71} 71}
72 72
73static inline bool forwarded_irq(struct irq_data *d)
74{
75 return d->handler_data != NULL;
76}
77
73static inline void __iomem *gic_dist_base(struct irq_data *d) 78static inline void __iomem *gic_dist_base(struct irq_data *d)
74{ 79{
75 if (gic_irq_in_rdist(d)) /* SGI+PPI -> SGI_base for this CPU */ 80 if (gic_irq_in_rdist(d)) /* SGI+PPI -> SGI_base for this CPU */
@@ -236,6 +241,16 @@ static void gic_mask_irq(struct irq_data *d)
236static void gic_eoimode1_mask_irq(struct irq_data *d) 241static void gic_eoimode1_mask_irq(struct irq_data *d)
237{ 242{
238 gic_mask_irq(d); 243 gic_mask_irq(d);
244 /*
245 * When masking a forwarded interrupt, make sure it is
246 * deactivated as well.
247 *
248 * This ensures that an interrupt that is getting
249 * disabled/masked will not get "stuck", because there is
250 * noone to deactivate it (guest is being terminated).
251 */
252 if (forwarded_irq(d))
253 gic_poke_irq(d, GICD_ICACTIVER);
239} 254}
240 255
241static void gic_unmask_irq(struct irq_data *d) 256static void gic_unmask_irq(struct irq_data *d)
@@ -306,9 +321,10 @@ static void gic_eoi_irq(struct irq_data *d)
306static void gic_eoimode1_eoi_irq(struct irq_data *d) 321static void gic_eoimode1_eoi_irq(struct irq_data *d)
307{ 322{
308 /* 323 /*
309 * No need to deactivate an LPI. 324 * No need to deactivate an LPI, or an interrupt that
325 * is is getting forwarded to a vcpu.
310 */ 326 */
311 if (gic_irq(d) >= 8192) 327 if (gic_irq(d) >= 8192 || forwarded_irq(d))
312 return; 328 return;
313 gic_write_dir(gic_irq(d)); 329 gic_write_dir(gic_irq(d));
314} 330}
@@ -339,6 +355,12 @@ static int gic_set_type(struct irq_data *d, unsigned int type)
339 return gic_configure_irq(irq, type, base, rwp_wait); 355 return gic_configure_irq(irq, type, base, rwp_wait);
340} 356}
341 357
358static int gic_irq_set_vcpu_affinity(struct irq_data *d, void *vcpu)
359{
360 d->handler_data = vcpu;
361 return 0;
362}
363
342static u64 gic_mpidr_to_affinity(u64 mpidr) 364static u64 gic_mpidr_to_affinity(u64 mpidr)
343{ 365{
344 u64 aff; 366 u64 aff;
@@ -703,6 +725,7 @@ static struct irq_chip gic_eoimode1_chip = {
703 .irq_set_affinity = gic_set_affinity, 725 .irq_set_affinity = gic_set_affinity,
704 .irq_get_irqchip_state = gic_irq_get_irqchip_state, 726 .irq_get_irqchip_state = gic_irq_get_irqchip_state,
705 .irq_set_irqchip_state = gic_irq_set_irqchip_state, 727 .irq_set_irqchip_state = gic_irq_set_irqchip_state,
728 .irq_set_vcpu_affinity = gic_irq_set_vcpu_affinity,
706 .flags = IRQCHIP_SET_TYPE_MASKED, 729 .flags = IRQCHIP_SET_TYPE_MASKED,
707}; 730};
708 731