diff options
Diffstat (limited to 'arch/x86')
-rw-r--r-- | arch/x86/include/asm/hw_irq.h | 2 | ||||
-rw-r--r-- | arch/x86/kernel/entry_64.S | 48 | ||||
-rw-r--r-- | arch/x86/kernel/irqinit_64.c | 66 |
3 files changed, 45 insertions, 71 deletions
diff --git a/arch/x86/include/asm/hw_irq.h b/arch/x86/include/asm/hw_irq.h index 27d33f92afe2..8de644b6b959 100644 --- a/arch/x86/include/asm/hw_irq.h +++ b/arch/x86/include/asm/hw_irq.h | |||
@@ -109,9 +109,7 @@ extern asmlinkage void smp_invalidate_interrupt(struct pt_regs *); | |||
109 | #endif | 109 | #endif |
110 | #endif | 110 | #endif |
111 | 111 | ||
112 | #ifdef CONFIG_X86_32 | ||
113 | extern void (*__initconst interrupt[NR_VECTORS-FIRST_EXTERNAL_VECTOR])(void); | 112 | extern void (*__initconst interrupt[NR_VECTORS-FIRST_EXTERNAL_VECTOR])(void); |
114 | #endif | ||
115 | 113 | ||
116 | typedef int vector_irq_t[NR_VECTORS]; | 114 | typedef int vector_irq_t[NR_VECTORS]; |
117 | DECLARE_PER_CPU(vector_irq_t, vector_irq); | 115 | DECLARE_PER_CPU(vector_irq_t, vector_irq); |
diff --git a/arch/x86/kernel/entry_64.S b/arch/x86/kernel/entry_64.S index b86f332c96a6..9b2aeaac9a6b 100644 --- a/arch/x86/kernel/entry_64.S +++ b/arch/x86/kernel/entry_64.S | |||
@@ -627,6 +627,46 @@ END(stub_rt_sigreturn) | |||
627 | vector already pushed) */ | 627 | vector already pushed) */ |
628 | #define XCPT_FRAME _frame ORIG_RAX | 628 | #define XCPT_FRAME _frame ORIG_RAX |
629 | 629 | ||
630 | /* | ||
631 | * Build the entry stubs and pointer table with some assembler magic. | ||
632 | * We pack 7 stubs into a single 32-byte chunk, which will fit in a | ||
633 | * single cache line on all modern x86 implementations. | ||
634 | */ | ||
635 | .section .init.rodata,"a" | ||
636 | ENTRY(interrupt) | ||
637 | .text | ||
638 | .p2align 5 | ||
639 | .p2align CONFIG_X86_L1_CACHE_SHIFT | ||
640 | ENTRY(irq_entries_start) | ||
641 | INTR_FRAME | ||
642 | vector=FIRST_EXTERNAL_VECTOR | ||
643 | .rept (NR_VECTORS-FIRST_EXTERNAL_VECTOR+6)/7 | ||
644 | .balign 32 | ||
645 | .rept 7 | ||
646 | .if vector < NR_VECTORS | ||
647 | .if vector != FIRST_EXTERNAL_VECTOR | ||
648 | CFI_ADJUST_CFA_OFFSET -8 | ||
649 | .endif | ||
650 | 1: pushq $(~vector+0x80) /* Note: always in signed byte range */ | ||
651 | CFI_ADJUST_CFA_OFFSET 8 | ||
652 | .if ((vector-FIRST_EXTERNAL_VECTOR)%7) != 6 | ||
653 | jmp 2f | ||
654 | .endif | ||
655 | .previous | ||
656 | .quad 1b | ||
657 | .text | ||
658 | vector=vector+1 | ||
659 | .endif | ||
660 | .endr | ||
661 | 2: jmp common_interrupt | ||
662 | .endr | ||
663 | CFI_ENDPROC | ||
664 | END(irq_entries_start) | ||
665 | |||
666 | .previous | ||
667 | END(interrupt) | ||
668 | .previous | ||
669 | |||
630 | /* | 670 | /* |
631 | * Interrupt entry/exit. | 671 | * Interrupt entry/exit. |
632 | * | 672 | * |
@@ -635,11 +675,12 @@ END(stub_rt_sigreturn) | |||
635 | * Entry runs with interrupts off. | 675 | * Entry runs with interrupts off. |
636 | */ | 676 | */ |
637 | 677 | ||
638 | /* 0(%rsp): interrupt number */ | 678 | /* 0(%rsp): ~(interrupt number)+0x80 */ |
639 | .macro interrupt func | 679 | .macro interrupt func |
680 | addq $-0x80,(%rsp) /* Adjust vector to [-256,-1] range */ | ||
640 | cld | 681 | cld |
641 | SAVE_ARGS | 682 | SAVE_ARGS |
642 | leaq -ARGOFFSET(%rsp),%rdi # arg1 for handler | 683 | leaq -ARGOFFSET(%rsp),%rdi /* arg1 for handler */ |
643 | pushq %rbp | 684 | pushq %rbp |
644 | /* | 685 | /* |
645 | * Save rbp twice: One is for marking the stack frame, as usual, and the | 686 | * Save rbp twice: One is for marking the stack frame, as usual, and the |
@@ -670,7 +711,8 @@ END(stub_rt_sigreturn) | |||
670 | call \func | 711 | call \func |
671 | .endm | 712 | .endm |
672 | 713 | ||
673 | ENTRY(common_interrupt) | 714 | .p2align CONFIG_X86_L1_CACHE_SHIFT |
715 | common_interrupt: | ||
674 | XCPT_FRAME | 716 | XCPT_FRAME |
675 | interrupt do_IRQ | 717 | interrupt do_IRQ |
676 | /* 0(%rsp): oldrsp-ARGOFFSET */ | 718 | /* 0(%rsp): oldrsp-ARGOFFSET */ |
diff --git a/arch/x86/kernel/irqinit_64.c b/arch/x86/kernel/irqinit_64.c index ff0235391285..8670b3ce626e 100644 --- a/arch/x86/kernel/irqinit_64.c +++ b/arch/x86/kernel/irqinit_64.c | |||
@@ -24,41 +24,6 @@ | |||
24 | #include <asm/i8259.h> | 24 | #include <asm/i8259.h> |
25 | 25 | ||
26 | /* | 26 | /* |
27 | * Common place to define all x86 IRQ vectors | ||
28 | * | ||
29 | * This builds up the IRQ handler stubs using some ugly macros in irq.h | ||
30 | * | ||
31 | * These macros create the low-level assembly IRQ routines that save | ||
32 | * register context and call do_IRQ(). do_IRQ() then does all the | ||
33 | * operations that are needed to keep the AT (or SMP IOAPIC) | ||
34 | * interrupt-controller happy. | ||
35 | */ | ||
36 | |||
37 | #define IRQ_NAME2(nr) nr##_interrupt(void) | ||
38 | #define IRQ_NAME(nr) IRQ_NAME2(IRQ##nr) | ||
39 | |||
40 | /* | ||
41 | * SMP has a few special interrupts for IPI messages | ||
42 | */ | ||
43 | |||
44 | #define BUILD_IRQ(nr) \ | ||
45 | asmlinkage void IRQ_NAME(nr); \ | ||
46 | asm("\n.text\n.p2align\n" \ | ||
47 | "IRQ" #nr "_interrupt:\n\t" \ | ||
48 | "push $~(" #nr ") ; " \ | ||
49 | "jmp common_interrupt\n" \ | ||
50 | ".previous"); | ||
51 | |||
52 | #define BI(x,y) \ | ||
53 | BUILD_IRQ(x##y) | ||
54 | |||
55 | #define BUILD_16_IRQS(x) \ | ||
56 | BI(x,0) BI(x,1) BI(x,2) BI(x,3) \ | ||
57 | BI(x,4) BI(x,5) BI(x,6) BI(x,7) \ | ||
58 | BI(x,8) BI(x,9) BI(x,a) BI(x,b) \ | ||
59 | BI(x,c) BI(x,d) BI(x,e) BI(x,f) | ||
60 | |||
61 | /* | ||
62 | * ISA PIC or low IO-APIC triggered (INTA-cycle or APIC) interrupts: | 27 | * ISA PIC or low IO-APIC triggered (INTA-cycle or APIC) interrupts: |
63 | * (these are usually mapped to vectors 0x30-0x3f) | 28 | * (these are usually mapped to vectors 0x30-0x3f) |
64 | */ | 29 | */ |
@@ -73,37 +38,6 @@ | |||
73 | * | 38 | * |
74 | * (these are usually mapped into the 0x30-0xff vector range) | 39 | * (these are usually mapped into the 0x30-0xff vector range) |
75 | */ | 40 | */ |
76 | BUILD_16_IRQS(0x2) BUILD_16_IRQS(0x3) | ||
77 | BUILD_16_IRQS(0x4) BUILD_16_IRQS(0x5) BUILD_16_IRQS(0x6) BUILD_16_IRQS(0x7) | ||
78 | BUILD_16_IRQS(0x8) BUILD_16_IRQS(0x9) BUILD_16_IRQS(0xa) BUILD_16_IRQS(0xb) | ||
79 | BUILD_16_IRQS(0xc) BUILD_16_IRQS(0xd) BUILD_16_IRQS(0xe) BUILD_16_IRQS(0xf) | ||
80 | |||
81 | #undef BUILD_16_IRQS | ||
82 | #undef BI | ||
83 | |||
84 | |||
85 | #define IRQ(x,y) \ | ||
86 | IRQ##x##y##_interrupt | ||
87 | |||
88 | #define IRQLIST_16(x) \ | ||
89 | IRQ(x,0), IRQ(x,1), IRQ(x,2), IRQ(x,3), \ | ||
90 | IRQ(x,4), IRQ(x,5), IRQ(x,6), IRQ(x,7), \ | ||
91 | IRQ(x,8), IRQ(x,9), IRQ(x,a), IRQ(x,b), \ | ||
92 | IRQ(x,c), IRQ(x,d), IRQ(x,e), IRQ(x,f) | ||
93 | |||
94 | /* for the irq vectors */ | ||
95 | static void (*__initdata interrupt[NR_VECTORS - FIRST_EXTERNAL_VECTOR])(void) = { | ||
96 | IRQLIST_16(0x2), IRQLIST_16(0x3), | ||
97 | IRQLIST_16(0x4), IRQLIST_16(0x5), IRQLIST_16(0x6), IRQLIST_16(0x7), | ||
98 | IRQLIST_16(0x8), IRQLIST_16(0x9), IRQLIST_16(0xa), IRQLIST_16(0xb), | ||
99 | IRQLIST_16(0xc), IRQLIST_16(0xd), IRQLIST_16(0xe), IRQLIST_16(0xf) | ||
100 | }; | ||
101 | |||
102 | #undef IRQ | ||
103 | #undef IRQLIST_16 | ||
104 | |||
105 | |||
106 | |||
107 | 41 | ||
108 | /* | 42 | /* |
109 | * IRQ2 is cascade interrupt to second interrupt controller | 43 | * IRQ2 is cascade interrupt to second interrupt controller |