2 * Interrupt descriptor table related code
4 * This file is licensed under the GPL V2
6 #include <linux/interrupt.h>
22 #define DEFAULT_STACK 0
24 #define G(_vector, _addr, _ist, _type, _dpl, _segment) \
32 .segment = _segment, \
36 #define INTG(_vector, _addr) \
37 G(_vector, _addr, DEFAULT_STACK, GATE_INTERRUPT, DPL0, __KERNEL_CS)
39 /* System interrupt gate */
40 #define SYSG(_vector, _addr) \
41 G(_vector, _addr, DEFAULT_STACK, GATE_INTERRUPT, DPL3, __KERNEL_CS)
43 /* Interrupt gate with interrupt stack */
44 #define ISTG(_vector, _addr, _ist) \
45 G(_vector, _addr, _ist, GATE_INTERRUPT, DPL0, __KERNEL_CS)
47 /* System interrupt gate with interrupt stack */
48 #define SISTG(_vector, _addr, _ist) \
49 G(_vector, _addr, _ist, GATE_INTERRUPT, DPL3, __KERNEL_CS)
52 #define TSKG(_vector, _gdt) \
53 G(_vector, NULL, DEFAULT_STACK, GATE_TASK, DPL0, _gdt << 3)
56 * Early traps running on the DEFAULT_STACK because the other interrupt
57 * stacks work only after cpu_init().
59 static const __initconst
struct idt_data early_idts
[] = {
60 INTG(X86_TRAP_DB
, debug
),
61 SYSG(X86_TRAP_BP
, int3
),
63 INTG(X86_TRAP_PF
, page_fault
),
68 * The default IDT entries which are set up in trap_init() before
69 * cpu_init() is invoked. Interrupt stacks cannot be used at that point and
70 * the traps which use them are reinitialized with IST after cpu_init() has
73 static const __initconst
struct idt_data def_idts
[] = {
74 INTG(X86_TRAP_DE
, divide_error
),
75 INTG(X86_TRAP_NMI
, nmi
),
76 INTG(X86_TRAP_BR
, bounds
),
77 INTG(X86_TRAP_UD
, invalid_op
),
78 INTG(X86_TRAP_NM
, device_not_available
),
79 INTG(X86_TRAP_OLD_MF
, coprocessor_segment_overrun
),
80 INTG(X86_TRAP_TS
, invalid_TSS
),
81 INTG(X86_TRAP_NP
, segment_not_present
),
82 INTG(X86_TRAP_SS
, stack_segment
),
83 INTG(X86_TRAP_GP
, general_protection
),
84 INTG(X86_TRAP_SPURIOUS
, spurious_interrupt_bug
),
85 INTG(X86_TRAP_MF
, coprocessor_error
),
86 INTG(X86_TRAP_AC
, alignment_check
),
87 INTG(X86_TRAP_XF
, simd_coprocessor_error
),
90 TSKG(X86_TRAP_DF
, GDT_ENTRY_DOUBLEFAULT_TSS
),
92 INTG(X86_TRAP_DF
, double_fault
),
94 INTG(X86_TRAP_DB
, debug
),
97 INTG(X86_TRAP_MC
, &machine_check
),
100 SYSG(X86_TRAP_OF
, overflow
),
101 #if defined(CONFIG_IA32_EMULATION)
102 SYSG(IA32_SYSCALL_VECTOR
, entry_INT80_compat
),
103 #elif defined(CONFIG_X86_32)
104 SYSG(IA32_SYSCALL_VECTOR
, entry_INT80_32
),
109 * The APIC and SMP idt entries
111 static const __initconst
struct idt_data apic_idts
[] = {
113 INTG(RESCHEDULE_VECTOR
, reschedule_interrupt
),
114 INTG(CALL_FUNCTION_VECTOR
, call_function_interrupt
),
115 INTG(CALL_FUNCTION_SINGLE_VECTOR
, call_function_single_interrupt
),
116 INTG(IRQ_MOVE_CLEANUP_VECTOR
, irq_move_cleanup_interrupt
),
117 INTG(REBOOT_VECTOR
, reboot_interrupt
),
120 #ifdef CONFIG_X86_THERMAL_VECTOR
121 INTG(THERMAL_APIC_VECTOR
, thermal_interrupt
),
124 #ifdef CONFIG_X86_MCE_THRESHOLD
125 INTG(THRESHOLD_APIC_VECTOR
, threshold_interrupt
),
128 #ifdef CONFIG_X86_MCE_AMD
129 INTG(DEFERRED_ERROR_VECTOR
, deferred_error_interrupt
),
132 #ifdef CONFIG_X86_LOCAL_APIC
133 INTG(LOCAL_TIMER_VECTOR
, apic_timer_interrupt
),
134 INTG(X86_PLATFORM_IPI_VECTOR
, x86_platform_ipi
),
135 # ifdef CONFIG_HAVE_KVM
136 INTG(POSTED_INTR_VECTOR
, kvm_posted_intr_ipi
),
137 INTG(POSTED_INTR_WAKEUP_VECTOR
, kvm_posted_intr_wakeup_ipi
),
138 INTG(POSTED_INTR_NESTED_VECTOR
, kvm_posted_intr_nested_ipi
),
140 # ifdef CONFIG_IRQ_WORK
141 INTG(IRQ_WORK_VECTOR
, irq_work_interrupt
),
143 INTG(SPURIOUS_APIC_VECTOR
, spurious_interrupt
),
144 INTG(ERROR_APIC_VECTOR
, error_interrupt
),
150 * Early traps running on the DEFAULT_STACK because the other interrupt
151 * stacks work only after cpu_init().
153 static const __initconst
struct idt_data early_pf_idts
[] = {
154 INTG(X86_TRAP_PF
, page_fault
),
158 * Override for the debug_idt. Same as the default, but with interrupt
159 * stack set to DEFAULT_STACK (0). Required for NMI trap handling.
161 static const __initconst
struct idt_data dbg_idts
[] = {
162 INTG(X86_TRAP_DB
, debug
),
163 INTG(X86_TRAP_BP
, int3
),
167 /* Must be page-aligned because the real IDT is used in a fixmap. */
168 gate_desc idt_table
[IDT_ENTRIES
] __page_aligned_bss
;
170 struct desc_ptr idt_descr __ro_after_init
= {
171 .size
= (IDT_ENTRIES
* 2 * sizeof(unsigned long)) - 1,
172 .address
= (unsigned long) idt_table
,
176 /* No need to be aligned, but done to keep all IDTs defined the same way. */
177 gate_desc debug_idt_table
[IDT_ENTRIES
] __page_aligned_bss
;
180 * The exceptions which use Interrupt stacks. They are setup after
181 * cpu_init() when the TSS has been initialized.
183 static const __initconst
struct idt_data ist_idts
[] = {
184 ISTG(X86_TRAP_DB
, debug
, DEBUG_STACK
),
185 ISTG(X86_TRAP_NMI
, nmi
, NMI_STACK
),
186 SISTG(X86_TRAP_BP
, int3
, DEBUG_STACK
),
187 ISTG(X86_TRAP_DF
, double_fault
, DOUBLEFAULT_STACK
),
188 #ifdef CONFIG_X86_MCE
189 ISTG(X86_TRAP_MC
, &machine_check
, MCE_STACK
),
194 * Override for the debug_idt. Same as the default, but with interrupt
195 * stack set to DEFAULT_STACK (0). Required for NMI trap handling.
197 const struct desc_ptr debug_idt_descr
= {
198 .size
= IDT_ENTRIES
* 16 - 1,
199 .address
= (unsigned long) debug_idt_table
,
203 static inline void idt_init_desc(gate_desc
*gate
, const struct idt_data
*d
)
205 unsigned long addr
= (unsigned long) d
->addr
;
207 gate
->offset_low
= (u16
) addr
;
208 gate
->segment
= (u16
) d
->segment
;
209 gate
->bits
= d
->bits
;
210 gate
->offset_middle
= (u16
) (addr
>> 16);
212 gate
->offset_high
= (u32
) (addr
>> 32);
218 idt_setup_from_table(gate_desc
*idt
, const struct idt_data
*t
, int size
, bool sys
)
222 for (; size
> 0; t
++, size
--) {
223 idt_init_desc(&desc
, t
);
224 write_idt_entry(idt
, t
->vector
, &desc
);
226 set_bit(t
->vector
, system_vectors
);
230 static void set_intr_gate(unsigned int n
, const void *addr
)
232 struct idt_data data
;
236 memset(&data
, 0, sizeof(data
));
239 data
.segment
= __KERNEL_CS
;
240 data
.bits
.type
= GATE_INTERRUPT
;
243 idt_setup_from_table(idt_table
, &data
, 1, false);
247 * idt_setup_early_traps - Initialize the idt table with early traps
249 * On X8664 these traps do not use interrupt stacks as they can't work
250 * before cpu_init() is invoked and sets up TSS. The IST variants are
251 * installed after that.
253 void __init
idt_setup_early_traps(void)
255 idt_setup_from_table(idt_table
, early_idts
, ARRAY_SIZE(early_idts
),
257 load_idt(&idt_descr
);
261 * idt_setup_traps - Initialize the idt table with default traps
263 void __init
idt_setup_traps(void)
265 idt_setup_from_table(idt_table
, def_idts
, ARRAY_SIZE(def_idts
), true);
270 * idt_setup_early_pf - Initialize the idt table with early pagefault handler
272 * On X8664 this does not use interrupt stacks as they can't work before
273 * cpu_init() is invoked and sets up TSS. The IST variant is installed
276 * FIXME: Why is 32bit and 64bit installing the PF handler at different
277 * places in the early setup code?
279 void __init
idt_setup_early_pf(void)
281 idt_setup_from_table(idt_table
, early_pf_idts
,
282 ARRAY_SIZE(early_pf_idts
), true);
286 * idt_setup_ist_traps - Initialize the idt table with traps using IST
288 void __init
idt_setup_ist_traps(void)
290 idt_setup_from_table(idt_table
, ist_idts
, ARRAY_SIZE(ist_idts
), true);
294 * idt_setup_debugidt_traps - Initialize the debug idt table with debug traps
296 void __init
idt_setup_debugidt_traps(void)
298 memcpy(&debug_idt_table
, &idt_table
, IDT_ENTRIES
* 16);
300 idt_setup_from_table(debug_idt_table
, dbg_idts
, ARRAY_SIZE(dbg_idts
), false);
305 * idt_setup_apic_and_irq_gates - Setup APIC/SMP and normal interrupt gates
307 void __init
idt_setup_apic_and_irq_gates(void)
309 int i
= FIRST_EXTERNAL_VECTOR
;
312 idt_setup_from_table(idt_table
, apic_idts
, ARRAY_SIZE(apic_idts
), true);
314 for_each_clear_bit_from(i
, system_vectors
, FIRST_SYSTEM_VECTOR
) {
315 entry
= irq_entries_start
+ 8 * (i
- FIRST_EXTERNAL_VECTOR
);
316 set_intr_gate(i
, entry
);
319 for_each_clear_bit_from(i
, system_vectors
, NR_VECTORS
) {
320 #ifdef CONFIG_X86_LOCAL_APIC
321 set_bit(i
, system_vectors
);
322 set_intr_gate(i
, spurious_interrupt
);
324 entry
= irq_entries_start
+ 8 * (i
- FIRST_EXTERNAL_VECTOR
);
325 set_intr_gate(i
, entry
);
331 * idt_setup_early_handler - Initializes the idt table with early handlers
333 void __init
idt_setup_early_handler(void)
337 for (i
= 0; i
< NUM_EXCEPTION_VECTORS
; i
++)
338 set_intr_gate(i
, early_idt_handler_array
[i
]);
340 for ( ; i
< NR_VECTORS
; i
++)
341 set_intr_gate(i
, early_ignore_irq
);
343 load_idt(&idt_descr
);
347 * idt_invalidate - Invalidate interrupt descriptor table
348 * @addr: The virtual address of the 'invalid' IDT
350 void idt_invalidate(void *addr
)
352 struct desc_ptr idt
= { .address
= (unsigned long) addr
, .size
= 0 };
357 void __init
update_intr_gate(unsigned int n
, const void *addr
)
359 if (WARN_ON_ONCE(!test_bit(n
, system_vectors
)))
361 set_intr_gate(n
, addr
);
364 void alloc_intr_gate(unsigned int n
, const void *addr
)
366 BUG_ON(n
< FIRST_SYSTEM_VECTOR
);
367 if (!test_and_set_bit(n
, system_vectors
))
368 set_intr_gate(n
, addr
);