diff options
-rw-r--r-- | arch/sparc/Kconfig | 1 | ||||
-rw-r--r-- | arch/sparc/kernel/irq_64.c | 32 |
2 files changed, 9 insertions, 24 deletions
diff --git a/arch/sparc/Kconfig b/arch/sparc/Kconfig index f766e6bf370e..9efd30f6a1ed 100644 --- a/arch/sparc/Kconfig +++ b/arch/sparc/Kconfig | |||
@@ -52,6 +52,7 @@ config SPARC64 | |||
52 | select PERF_USE_VMALLOC | 52 | select PERF_USE_VMALLOC |
53 | select HAVE_GENERIC_HARDIRQS | 53 | select HAVE_GENERIC_HARDIRQS |
54 | select GENERIC_HARDIRQS_NO_DEPRECATED | 54 | select GENERIC_HARDIRQS_NO_DEPRECATED |
55 | select IRQ_PREFLOW_FASTEOI | ||
55 | 56 | ||
56 | config ARCH_DEFCONFIG | 57 | config ARCH_DEFCONFIG |
57 | string | 58 | string |
diff --git a/arch/sparc/kernel/irq_64.c b/arch/sparc/kernel/irq_64.c index eb16e3b8a2dd..3c8b2666c325 100644 --- a/arch/sparc/kernel/irq_64.c +++ b/arch/sparc/kernel/irq_64.c | |||
@@ -344,10 +344,6 @@ static void sun4u_irq_disable(struct irq_data *data) | |||
344 | static void sun4u_irq_eoi(struct irq_data *data) | 344 | static void sun4u_irq_eoi(struct irq_data *data) |
345 | { | 345 | { |
346 | struct irq_handler_data *handler_data = data->handler_data; | 346 | struct irq_handler_data *handler_data = data->handler_data; |
347 | struct irq_desc *desc = irq_desc + data->irq; | ||
348 | |||
349 | if (unlikely(desc->status & (IRQ_DISABLED|IRQ_INPROGRESS))) | ||
350 | return; | ||
351 | 347 | ||
352 | if (likely(handler_data)) | 348 | if (likely(handler_data)) |
353 | upa_writeq(ICLR_IDLE, handler_data->iclr); | 349 | upa_writeq(ICLR_IDLE, handler_data->iclr); |
@@ -402,12 +398,8 @@ static void sun4v_irq_disable(struct irq_data *data) | |||
402 | static void sun4v_irq_eoi(struct irq_data *data) | 398 | static void sun4v_irq_eoi(struct irq_data *data) |
403 | { | 399 | { |
404 | unsigned int ino = irq_table[data->irq].dev_ino; | 400 | unsigned int ino = irq_table[data->irq].dev_ino; |
405 | struct irq_desc *desc = irq_desc + data->irq; | ||
406 | int err; | 401 | int err; |
407 | 402 | ||
408 | if (unlikely(desc->status & (IRQ_DISABLED|IRQ_INPROGRESS))) | ||
409 | return; | ||
410 | |||
411 | err = sun4v_intr_setstate(ino, HV_INTR_STATE_IDLE); | 403 | err = sun4v_intr_setstate(ino, HV_INTR_STATE_IDLE); |
412 | if (err != HV_EOK) | 404 | if (err != HV_EOK) |
413 | printk(KERN_ERR "sun4v_intr_setstate(%x): " | 405 | printk(KERN_ERR "sun4v_intr_setstate(%x): " |
@@ -481,13 +473,9 @@ static void sun4v_virq_disable(struct irq_data *data) | |||
481 | 473 | ||
482 | static void sun4v_virq_eoi(struct irq_data *data) | 474 | static void sun4v_virq_eoi(struct irq_data *data) |
483 | { | 475 | { |
484 | struct irq_desc *desc = irq_desc + data->irq; | ||
485 | unsigned long dev_handle, dev_ino; | 476 | unsigned long dev_handle, dev_ino; |
486 | int err; | 477 | int err; |
487 | 478 | ||
488 | if (unlikely(desc->status & (IRQ_DISABLED|IRQ_INPROGRESS))) | ||
489 | return; | ||
490 | |||
491 | dev_handle = irq_table[data->irq].dev_handle; | 479 | dev_handle = irq_table[data->irq].dev_handle; |
492 | dev_ino = irq_table[data->irq].dev_ino; | 480 | dev_ino = irq_table[data->irq].dev_ino; |
493 | 481 | ||
@@ -505,6 +493,7 @@ static struct irq_chip sun4u_irq = { | |||
505 | .irq_disable = sun4u_irq_disable, | 493 | .irq_disable = sun4u_irq_disable, |
506 | .irq_eoi = sun4u_irq_eoi, | 494 | .irq_eoi = sun4u_irq_eoi, |
507 | .irq_set_affinity = sun4u_set_affinity, | 495 | .irq_set_affinity = sun4u_set_affinity, |
496 | .flags = IRQCHIP_EOI_IF_HANDLED, | ||
508 | }; | 497 | }; |
509 | 498 | ||
510 | static struct irq_chip sun4v_irq = { | 499 | static struct irq_chip sun4v_irq = { |
@@ -513,6 +502,7 @@ static struct irq_chip sun4v_irq = { | |||
513 | .irq_disable = sun4v_irq_disable, | 502 | .irq_disable = sun4v_irq_disable, |
514 | .irq_eoi = sun4v_irq_eoi, | 503 | .irq_eoi = sun4v_irq_eoi, |
515 | .irq_set_affinity = sun4v_set_affinity, | 504 | .irq_set_affinity = sun4v_set_affinity, |
505 | .flags = IRQCHIP_EOI_IF_HANDLED, | ||
516 | }; | 506 | }; |
517 | 507 | ||
518 | static struct irq_chip sun4v_virq = { | 508 | static struct irq_chip sun4v_virq = { |
@@ -521,16 +511,15 @@ static struct irq_chip sun4v_virq = { | |||
521 | .irq_disable = sun4v_virq_disable, | 511 | .irq_disable = sun4v_virq_disable, |
522 | .irq_eoi = sun4v_virq_eoi, | 512 | .irq_eoi = sun4v_virq_eoi, |
523 | .irq_set_affinity = sun4v_virt_set_affinity, | 513 | .irq_set_affinity = sun4v_virt_set_affinity, |
514 | .flags = IRQCHIP_EOI_IF_HANDLED, | ||
524 | }; | 515 | }; |
525 | 516 | ||
526 | static void pre_flow_handler(unsigned int irq, struct irq_desc *desc) | 517 | static void pre_flow_handler(struct irq_data *d) |
527 | { | 518 | { |
528 | struct irq_handler_data *handler_data = get_irq_data(irq); | 519 | struct irq_handler_data *handler_data = irq_data_get_irq_handler_data(d); |
529 | unsigned int ino = irq_table[irq].dev_ino; | 520 | unsigned int ino = irq_table[d->irq].dev_ino; |
530 | 521 | ||
531 | handler_data->pre_handler(ino, handler_data->arg1, handler_data->arg2); | 522 | handler_data->pre_handler(ino, handler_data->arg1, handler_data->arg2); |
532 | |||
533 | handle_fasteoi_irq(irq, desc); | ||
534 | } | 523 | } |
535 | 524 | ||
536 | void irq_install_pre_handler(int irq, | 525 | void irq_install_pre_handler(int irq, |
@@ -538,13 +527,12 @@ void irq_install_pre_handler(int irq, | |||
538 | void *arg1, void *arg2) | 527 | void *arg1, void *arg2) |
539 | { | 528 | { |
540 | struct irq_handler_data *handler_data = get_irq_data(irq); | 529 | struct irq_handler_data *handler_data = get_irq_data(irq); |
541 | struct irq_desc *desc = irq_desc + irq; | ||
542 | 530 | ||
543 | handler_data->pre_handler = func; | 531 | handler_data->pre_handler = func; |
544 | handler_data->arg1 = arg1; | 532 | handler_data->arg1 = arg1; |
545 | handler_data->arg2 = arg2; | 533 | handler_data->arg2 = arg2; |
546 | 534 | ||
547 | desc->handle_irq = pre_flow_handler; | 535 | __irq_set_preflow_handler(irq, pre_flow_handler); |
548 | } | 536 | } |
549 | 537 | ||
550 | unsigned int build_irq(int inofixup, unsigned long iclr, unsigned long imap) | 538 | unsigned int build_irq(int inofixup, unsigned long iclr, unsigned long imap) |
@@ -734,7 +722,6 @@ void __irq_entry handler_irq(int pil, struct pt_regs *regs) | |||
734 | orig_sp = set_hardirq_stack(); | 722 | orig_sp = set_hardirq_stack(); |
735 | 723 | ||
736 | while (bucket_pa) { | 724 | while (bucket_pa) { |
737 | struct irq_desc *desc; | ||
738 | unsigned long next_pa; | 725 | unsigned long next_pa; |
739 | unsigned int irq; | 726 | unsigned int irq; |
740 | 727 | ||
@@ -742,10 +729,7 @@ void __irq_entry handler_irq(int pil, struct pt_regs *regs) | |||
742 | irq = bucket_get_irq(bucket_pa); | 729 | irq = bucket_get_irq(bucket_pa); |
743 | bucket_clear_chain_pa(bucket_pa); | 730 | bucket_clear_chain_pa(bucket_pa); |
744 | 731 | ||
745 | desc = irq_desc + irq; | 732 | generic_handle_irq(irq); |
746 | |||
747 | if (!(desc->status & IRQ_DISABLED)) | ||
748 | desc->handle_irq(irq, desc); | ||
749 | 733 | ||
750 | bucket_pa = next_pa; | 734 | bucket_pa = next_pa; |
751 | } | 735 | } |