summaryrefslogtreecommitdiffstats
path: root/kernel/irq/manage.c
diff options
context:
space:
mode:
authorMarc Zyngier <marc.zyngier@arm.com>2017-06-23 16:42:57 -0400
committerMarc Zyngier <marc.zyngier@arm.com>2017-08-23 06:09:14 -0400
commit0abce64a55ae44d39b92f8e672736f4f324e610f (patch)
tree8a95c64171ac618a93e2b93421d4ef86f370e5cc /kernel/irq/manage.c
parente81f54c668d89e50bad38f3fc4c5ea6e4be3a96e (diff)
genirq: Let irq_set_vcpu_affinity() iterate over hierarchy
When assigning an interrupt to a vcpu, it is not unlikely that the level of the hierarchy implementing irq_set_vcpu_affinity is not the top level (think a generic MSI domain on top of a virtualization aware interrupt controller). In such a case, let's iterate over the hierarchy until we find an irqchip implementing it. Reviewed-by: Thomas Gleixner <tglx@linutronix.de> Signed-off-by: Marc Zyngier <marc.zyngier@arm.com>
Diffstat (limited to 'kernel/irq/manage.c')
-rw-r--r--kernel/irq/manage.c14
1 files changed, 12 insertions, 2 deletions
diff --git a/kernel/irq/manage.c b/kernel/irq/manage.c
index 1d1a5b945ab4..573dc52b0806 100644
--- a/kernel/irq/manage.c
+++ b/kernel/irq/manage.c
@@ -400,8 +400,18 @@ int irq_set_vcpu_affinity(unsigned int irq, void *vcpu_info)
400 return -EINVAL; 400 return -EINVAL;
401 401
402 data = irq_desc_get_irq_data(desc); 402 data = irq_desc_get_irq_data(desc);
403 chip = irq_data_get_irq_chip(data); 403 do {
404 if (chip && chip->irq_set_vcpu_affinity) 404 chip = irq_data_get_irq_chip(data);
405 if (chip && chip->irq_set_vcpu_affinity)
406 break;
407#ifdef CONFIG_IRQ_DOMAIN_HIERARCHY
408 data = data->parent_data;
409#else
410 data = NULL;
411#endif
412 } while (data);
413
414 if (data)
405 ret = chip->irq_set_vcpu_affinity(data, vcpu_info); 415 ret = chip->irq_set_vcpu_affinity(data, vcpu_info);
406 irq_put_desc_unlock(desc, flags); 416 irq_put_desc_unlock(desc, flags);
407 417