diff options
author | Marc Zyngier <marc.zyngier@arm.com> | 2017-06-23 16:42:57 -0400 |
---|---|---|
committer | Marc Zyngier <marc.zyngier@arm.com> | 2017-08-23 06:09:14 -0400 |
commit | 0abce64a55ae44d39b92f8e672736f4f324e610f (patch) | |
tree | 8a95c64171ac618a93e2b93421d4ef86f370e5cc /kernel/irq/manage.c | |
parent | e81f54c668d89e50bad38f3fc4c5ea6e4be3a96e (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.c | 14 |
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 | ||