diff options
author | H. Peter Anvin <hpa@zytor.com> | 2008-11-11 16:51:52 -0500 |
---|---|---|
committer | H. Peter Anvin <hpa@zytor.com> | 2008-11-11 16:51:52 -0500 |
commit | 939b787130bf22887a09d8fd2641a094dcef8c22 (patch) | |
tree | 6bdd272bb742bf2916d35c04cb8a6dd24e2dd135 /arch/x86/kernel | |
parent | b7c6244f13d37592003b46e12500a90e9781ad9d (diff) |
x86: 64 bits: shrink and align IRQ stubs
Move the IRQ stub generation to assembly to simplify it and for
consistency with 32 bits. Doing it in a C file with asm() statements
doesn't help clarity, and it prevents some optimizations.
Shrink the IRQ stubs down to just over four bytes per (we fit seven
into a 32-byte chunk.) This shrinks the total icache consumption of
the IRQ stubs down to an even kilobyte, if all of them are in active
use.
The downside is that we end up with a double jump, which could have a
negative effect on some pipelines. The double jump is always inside
the same cacheline on any modern chips.
To get the most effect, cache-align the IRQ stubs.
This makes the 64-bit code match changes already done to the 32-bit
code, and should open up irqinit*.c for unification.
Signed-off-by: H. Peter Anvin <hpa@zytor.com>
Diffstat (limited to 'arch/x86/kernel')
-rw-r--r-- | arch/x86/kernel/entry_64.S | 48 | ||||
-rw-r--r-- | arch/x86/kernel/irqinit_64.c | 66 |
2 files changed, 45 insertions, 69 deletions
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 |