xen: events: propagate irq allocation failure instead of panicking
Running out of IRQs need not be fatal to the machine as a whole. Signed-off-by: Ian Campbell <ian.campbell@citrix.com> Cc: Jeremy Fitzhardinge <jeremy@goop.org> Signed-off-by: Konrad Rzeszutek Wilk <konrad.wilk@oracle.com>
This commit is contained in:
Родитель
6cb9bf3aaf
Коммит
7bee976822
|
@ -406,7 +406,7 @@ static void xen_irq_init(unsigned irq)
|
||||||
list_add_tail(&info->list, &xen_irq_list_head);
|
list_add_tail(&info->list, &xen_irq_list_head);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int xen_allocate_irq_dynamic(void)
|
static int __must_check xen_allocate_irq_dynamic(void)
|
||||||
{
|
{
|
||||||
int first = 0;
|
int first = 0;
|
||||||
int irq;
|
int irq;
|
||||||
|
@ -425,15 +425,12 @@ static int xen_allocate_irq_dynamic(void)
|
||||||
|
|
||||||
irq = irq_alloc_desc_from(first, -1);
|
irq = irq_alloc_desc_from(first, -1);
|
||||||
|
|
||||||
if (irq < 0)
|
|
||||||
panic("No available IRQ to bind to: increase nr_irqs!\n");
|
|
||||||
|
|
||||||
xen_irq_init(irq);
|
xen_irq_init(irq);
|
||||||
|
|
||||||
return irq;
|
return irq;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int xen_allocate_irq_gsi(unsigned gsi)
|
static int __must_check xen_allocate_irq_gsi(unsigned gsi)
|
||||||
{
|
{
|
||||||
int irq;
|
int irq;
|
||||||
|
|
||||||
|
@ -452,9 +449,6 @@ static int xen_allocate_irq_gsi(unsigned gsi)
|
||||||
else
|
else
|
||||||
irq = irq_alloc_desc_at(gsi, -1);
|
irq = irq_alloc_desc_at(gsi, -1);
|
||||||
|
|
||||||
if (irq < 0)
|
|
||||||
panic("Unable to allocate to IRQ%d (%d)\n", gsi, irq);
|
|
||||||
|
|
||||||
xen_irq_init(irq);
|
xen_irq_init(irq);
|
||||||
|
|
||||||
return irq;
|
return irq;
|
||||||
|
@ -640,6 +634,8 @@ int xen_bind_pirq_gsi_to_irq(unsigned gsi,
|
||||||
}
|
}
|
||||||
|
|
||||||
irq = xen_allocate_irq_gsi(gsi);
|
irq = xen_allocate_irq_gsi(gsi);
|
||||||
|
if (irq < 0)
|
||||||
|
goto out;
|
||||||
|
|
||||||
set_irq_chip_and_handler_name(irq, &xen_pirq_chip,
|
set_irq_chip_and_handler_name(irq, &xen_pirq_chip,
|
||||||
handle_level_irq, name);
|
handle_level_irq, name);
|
||||||
|
@ -771,6 +767,8 @@ int bind_evtchn_to_irq(unsigned int evtchn)
|
||||||
|
|
||||||
if (irq == -1) {
|
if (irq == -1) {
|
||||||
irq = xen_allocate_irq_dynamic();
|
irq = xen_allocate_irq_dynamic();
|
||||||
|
if (irq == -1)
|
||||||
|
goto out;
|
||||||
|
|
||||||
set_irq_chip_and_handler_name(irq, &xen_dynamic_chip,
|
set_irq_chip_and_handler_name(irq, &xen_dynamic_chip,
|
||||||
handle_fasteoi_irq, "event");
|
handle_fasteoi_irq, "event");
|
||||||
|
@ -778,6 +776,7 @@ int bind_evtchn_to_irq(unsigned int evtchn)
|
||||||
xen_irq_info_evtchn_init(irq, evtchn);
|
xen_irq_info_evtchn_init(irq, evtchn);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
out:
|
||||||
spin_unlock(&irq_mapping_update_lock);
|
spin_unlock(&irq_mapping_update_lock);
|
||||||
|
|
||||||
return irq;
|
return irq;
|
||||||
|
@ -829,6 +828,8 @@ int bind_virq_to_irq(unsigned int virq, unsigned int cpu)
|
||||||
|
|
||||||
if (irq == -1) {
|
if (irq == -1) {
|
||||||
irq = xen_allocate_irq_dynamic();
|
irq = xen_allocate_irq_dynamic();
|
||||||
|
if (irq == -1)
|
||||||
|
goto out;
|
||||||
|
|
||||||
set_irq_chip_and_handler_name(irq, &xen_percpu_chip,
|
set_irq_chip_and_handler_name(irq, &xen_percpu_chip,
|
||||||
handle_percpu_irq, "virq");
|
handle_percpu_irq, "virq");
|
||||||
|
@ -845,6 +846,7 @@ int bind_virq_to_irq(unsigned int virq, unsigned int cpu)
|
||||||
bind_evtchn_to_cpu(evtchn, cpu);
|
bind_evtchn_to_cpu(evtchn, cpu);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
out:
|
||||||
spin_unlock(&irq_mapping_update_lock);
|
spin_unlock(&irq_mapping_update_lock);
|
||||||
|
|
||||||
return irq;
|
return irq;
|
||||||
|
@ -897,6 +899,8 @@ int bind_evtchn_to_irqhandler(unsigned int evtchn,
|
||||||
int retval;
|
int retval;
|
||||||
|
|
||||||
irq = bind_evtchn_to_irq(evtchn);
|
irq = bind_evtchn_to_irq(evtchn);
|
||||||
|
if (irq < 0)
|
||||||
|
return irq;
|
||||||
retval = request_irq(irq, handler, irqflags, devname, dev_id);
|
retval = request_irq(irq, handler, irqflags, devname, dev_id);
|
||||||
if (retval != 0) {
|
if (retval != 0) {
|
||||||
unbind_from_irq(irq);
|
unbind_from_irq(irq);
|
||||||
|
@ -915,6 +919,8 @@ int bind_virq_to_irqhandler(unsigned int virq, unsigned int cpu,
|
||||||
int retval;
|
int retval;
|
||||||
|
|
||||||
irq = bind_virq_to_irq(virq, cpu);
|
irq = bind_virq_to_irq(virq, cpu);
|
||||||
|
if (irq < 0)
|
||||||
|
return irq;
|
||||||
retval = request_irq(irq, handler, irqflags, devname, dev_id);
|
retval = request_irq(irq, handler, irqflags, devname, dev_id);
|
||||||
if (retval != 0) {
|
if (retval != 0) {
|
||||||
unbind_from_irq(irq);
|
unbind_from_irq(irq);
|
||||||
|
|
Загрузка…
Ссылка в новой задаче